ce426f
Adds --install-langs support build-locale-archive.
ce426f
ce426f
commit 0457f649e3fe6299efe384da13dfc923bbe65707
ce426f
Author: Carlos O'Donell <carlos@systemhalted.org>
ce426f
Date:   Thu Sep 17 12:24:49 2015 -0400
ce426f
ce426f
[snip]
ce426f
    - Fix --install-langs bug which causes SIGABRT (#1262040).
ce426f
ce426f
commit 91764bd9ec690d4b8a886c0a3a104aac12d340d2
ce426f
Author: Carlos O'Donell <carlos@systemhalted.org>
ce426f
Date:   Thu Mar 5 16:05:43 2015 -0500
ce426f
ce426f
    Resolves: #156477
ce426f
    
ce426f
    - Support installing only those locales specified by the RPM macro
ce426f
      %%_install_langs (#156477).
ce426f
ce426f
Index: glibc-2.17-c758a686/releng/build-locale-archive.c
ce426f
===================================================================
ce426f
--- glibc-2.17-c758a686.orig/releng/build-locale-archive.c
ce426f
+++ glibc-2.17-c758a686/releng/build-locale-archive.c
ce426f
@@ -8,6 +8,7 @@
ce426f
 #include <stdbool.h>
ce426f
 #include <stdio.h>
ce426f
 #include <stdlib.h>
ce426f
+#include <getopt.h>
ce426f
 #include <string.h>
ce426f
 #include <sys/mman.h>
ce426f
 #include <sys/stat.h>
ce426f
@@ -21,6 +22,7 @@ const char *alias_file = DATADIR "/local
ce426f
 const char *locar_file = PREFIX "/lib/locale/locale-archive";
ce426f
 const char *tmpl_file = PREFIX "/lib/locale/locale-archive.tmpl";
ce426f
 const char *loc_path = PREFIX "/lib/locale/";
ce426f
+/* Flags set by `--verbose` option.  */
ce426f
 int be_quiet = 1;
ce426f
 int verbose = 0;
ce426f
 int max_locarchive_open_retry = 10;
ce426f
@@ -96,7 +98,7 @@ open_tmpl_archive (struct locarhandle *a
ce426f
   struct stat64 st;
ce426f
   int fd;
ce426f
   struct locarhead head;
ce426f
-  const char *archivefname = tmpl_file;
ce426f
+  const char *archivefname = ah->fname == NULL ? tmpl_file : ah->fname;
ce426f
 
ce426f
   /* Open the archive.  We must have exclusive write access.  */
ce426f
   fd = open64 (archivefname, O_RDONLY);
ce426f
@@ -116,7 +118,7 @@ open_tmpl_archive (struct locarhandle *a
ce426f
   ah->mmaped = (head.sumhash_offset
ce426f
 		+ head.sumhash_size * sizeof (struct sumhashent));
ce426f
   if (ah->mmaped > (unsigned long) st.st_size)
ce426f
-    error (EXIT_FAILURE, 0, "locale archite template file truncated");
ce426f
+    error (EXIT_FAILURE, 0, "locale archive template file truncated");
ce426f
   ah->mmaped = st.st_size;
ce426f
   ah->reserved = st.st_size;
ce426f
 
ce426f
@@ -250,7 +252,10 @@ compute_data (struct locarhandle *ah, st
ce426f
 }
ce426f
 
ce426f
 static int
ce426f
-fill_archive (struct locarhandle *tmpl_ah, size_t nlist, char *list[],
ce426f
+fill_archive (struct locarhandle *tmpl_ah,
ce426f
+	      const char *fname,
ce426f
+	      size_t install_langs_count, char *install_langs_list[],
ce426f
+	      size_t nlist, char *list[],
ce426f
 	      const char *primary)
ce426f
 {
ce426f
   struct locarhandle ah;
ce426f
@@ -281,11 +286,40 @@ fill_archive (struct locarhandle *tmpl_a
ce426f
   for (cnt = used = 0; cnt < head->namehash_size; ++cnt)
ce426f
     if (namehashtab[cnt].locrec_offset != 0)
ce426f
       {
ce426f
+	char * name;
ce426f
+	int i;
ce426f
 	assert (used < head->namehash_used);
ce426f
-	names[used].name = tmpl_ah->addr + namehashtab[cnt].name_offset;
ce426f
-	names[used++].locrec
ce426f
-	  = (struct locrecent *) ((char *) tmpl_ah->addr +
ce426f
-				  namehashtab[cnt].locrec_offset);
ce426f
+        name = tmpl_ah->addr + namehashtab[cnt].name_offset;
ce426f
+        if (install_langs_count == 0)
ce426f
+          {
ce426f
+	    /* Always intstall the entry.  */
ce426f
+            names[used].name = name;
ce426f
+            names[used++].locrec
ce426f
+                = (struct locrecent *) ((char *) tmpl_ah->addr +
ce426f
+                                        namehashtab[cnt].locrec_offset);
ce426f
+          }
ce426f
+        else
ce426f
+          {
ce426f
+	    /* Only install the entry if the user asked for it via
ce426f
+	       --install-langs.  */
ce426f
+            for (i = 0; i < install_langs_count; i++)
ce426f
+              {
ce426f
+		/* Add one for "_" and one for the null terminator.  */
ce426f
+		size_t len = strlen (install_langs_list[i]) + 2;
ce426f
+		char *install_lang = (char *)xmalloc (len);
ce426f
+                strcpy (install_lang, install_langs_list[i]);
ce426f
+                if (strchr (install_lang, '_') == NULL)
ce426f
+                  strcat (install_lang, "_");
ce426f
+                if (strncmp (name, install_lang, strlen (install_lang)) == 0)
ce426f
+                  {
ce426f
+                    names[used].name = name;
ce426f
+                    names[used++].locrec
ce426f
+		      = (struct locrecent *) ((char *)tmpl_ah->addr
ce426f
+					      + namehashtab[cnt].locrec_offset);
ce426f
+                  }
ce426f
+		free (install_lang);
ce426f
+              }
ce426f
+          }
ce426f
       }
ce426f
 
ce426f
   /* Sort the names.  */
ce426f
@@ -304,6 +338,9 @@ fill_archive (struct locarhandle *tmpl_a
ce426f
 
ce426f
   /* Open the archive.  This call never returns if we cannot
ce426f
      successfully open the archive.  */
ce426f
+  ah.fname = NULL;
ce426f
+  if (fname != NULL)
ce426f
+    ah.fname = fname;
ce426f
   open_archive (&ah, false);
ce426f
 
ce426f
   if (primary != NULL)
ce426f
@@ -532,24 +569,197 @@ fill_archive (struct locarhandle *tmpl_a
ce426f
   return result;
ce426f
 }
ce426f
 
ce426f
-int main ()
ce426f
+void usage()
ce426f
+{
ce426f
+  printf ("\
ce426f
+Usage: build-locale-archive [OPTION]... [TEMPLATE-FILE] [ARCHIVE-FILE]\n\
ce426f
+ Builds a locale archive from a template file.\n\
ce426f
+ Options:\n\
ce426f
+  -h, --help                 Print this usage message.\n\
ce426f
+  -v, --verbose              Verbose execution.\n\
ce426f
+  -l, --install-langs=LIST   Only include locales given in LIST into the \n\
ce426f
+                             locale archive.  LIST is a colon separated list\n\
ce426f
+                             of locale prefixes, for example \"de:en:ja\".\n\
ce426f
+                             The special argument \"all\" means to install\n\
ce426f
+                             all languages and it must be present by itself.\n\
ce426f
+                             If \"all\" is present with any other language it\n\
ce426f
+                             will be treated as the name of a locale.\n\
ce426f
+                             If the --install-langs option is missing, all\n\
ce426f
+                             locales are installed. The colon separated list\n\
ce426f
+                             can contain any strings matching the beginning of\n\
ce426f
+                             locale names.\n\
ce426f
+                             If a string does not contain a \"_\", it is added.\n\
ce426f
+                             Examples:\n\
ce426f
+                               --install-langs=\"en\"\n\
ce426f
+                                 installs en_US, en_US.iso88591,\n\
ce426f
+                                 en_US.iso885915, en_US.utf8,\n\
ce426f
+                                 en_GB ...\n\
ce426f
+                               --install-langs=\"en_US.utf8\"\n\
ce426f
+                                 installs only en_US.utf8.\n\
ce426f
+                               --install-langs=\"ko\"\n\
ce426f
+                                 installs ko_KR, ko_KR.euckr,\n\
ce426f
+                                 ko_KR.utf8 but *not* kok_IN\n\
ce426f
+                                 because \"ko\" does not contain\n\
ce426f
+                                 \"_\" and it is silently added\n\
ce426f
+                               --install-langs\"ko:kok\"\n\
ce426f
+                                 installs ko_KR, ko_KR.euckr,\n\
ce426f
+                                 ko_KR.utf8, kok_IN, and\n\
ce426f
+                                 kok_IN.utf8.\n\
ce426f
+                               --install-langs=\"POSIX\" will\n\
ce426f
+                                 installs *no* locales at all\n\
ce426f
+                                 because POSIX matches none of\n\
ce426f
+                                 the locales. Actually, any string\n\
ce426f
+                                 matching nothing will do that.\n\
ce426f
+                                 POSIX and C will always be\n\
ce426f
+                                 available because they are\n\
ce426f
+                                 builtin.\n\
ce426f
+                             Aliases are installed as well,\n\
ce426f
+                             i.e. --install-langs=\"de\"\n\
ce426f
+                             will install not only every locale starting with\n\
ce426f
+                             \"de\" but also the aliases \"deutsch\"\n\
ce426f
+                             and and \"german\" although the latter does not\n\
ce426f
+                             start with \"de\".\n\
ce426f
+\n\
ce426f
+  If the arguments TEMPLATE-FILE and ARCHIVE-FILE are not given the locations\n\
ce426f
+  where the glibc used expects these files are used by default.\n\
ce426f
+");
ce426f
+}
ce426f
+
ce426f
+int main (int argc, char *argv[])
ce426f
 {
ce426f
   char path[4096];
ce426f
   DIR *dirp;
ce426f
   struct dirent64 *d;
ce426f
   struct stat64 st;
ce426f
   char *list[16384], *primary;
ce426f
+  char *lang;
ce426f
+  int install_langs_count = 0;
ce426f
+  int i;
ce426f
+  char *install_langs_arg, *ila_start;
ce426f
+  char **install_langs_list;
ce426f
   unsigned int cnt = 0;
ce426f
   struct locarhandle tmpl_ah;
ce426f
+  char *new_locar_fname = NULL;
ce426f
   size_t loc_path_len = strlen (loc_path);
ce426f
 
ce426f
+  while (1)
ce426f
+    {
ce426f
+      int c;
ce426f
+
ce426f
+      static struct option long_options[] =
ce426f
+        {
ce426f
+            {"help",            no_argument,       0, 'h'},
ce426f
+            {"verbose",         no_argument,       0, 'v'},
ce426f
+            {"install-langs",   required_argument, 0, 'l'},
ce426f
+            {0, 0, 0, 0}
ce426f
+        };
ce426f
+      /* getopt_long stores the option index here. */
ce426f
+      int option_index = 0;
ce426f
+
ce426f
+      c = getopt_long (argc, argv, "vhl:",
ce426f
+                       long_options, &option_index);
ce426f
+
ce426f
+      /* Detect the end of the options. */
ce426f
+      if (c == -1)
ce426f
+        break;
ce426f
+
ce426f
+      switch (c)
ce426f
+        {
ce426f
+        case 0:
ce426f
+          printf ("unknown option %s", long_options[option_index].name);
ce426f
+          if (optarg)
ce426f
+            printf (" with arg %s", optarg);
ce426f
+          printf ("\n");
ce426f
+          usage ();
ce426f
+          exit (1);
ce426f
+
ce426f
+        case 'v':
ce426f
+          verbose = 1;
ce426f
+          be_quiet = 0;
ce426f
+          break;
ce426f
+
ce426f
+        case 'h':
ce426f
+          usage ();
ce426f
+          exit (0);
ce426f
+
ce426f
+        case 'l':
ce426f
+          install_langs_arg = ila_start = strdup (optarg);
ce426f
+          /* If the argument to --install-lang is "all", do
ce426f
+             not limit the list of languages to install and install
ce426f
+             them all.  We do not support installing a single locale
ce426f
+	     called "all".  */
ce426f
+#define MAGIC_INSTALL_ALL "all"
ce426f
+          if (install_langs_arg != NULL
ce426f
+	      && install_langs_arg[0] != '\0'
ce426f
+	      && !(strncmp(install_langs_arg, MAGIC_INSTALL_ALL,
ce426f
+			   strlen(MAGIC_INSTALL_ALL)) == 0
ce426f
+		   && strlen (install_langs_arg) == 3))
ce426f
+            {
ce426f
+	      /* Count the number of languages we will install.  */
ce426f
+              while (true)
ce426f
+                {
ce426f
+                  lang = strtok(install_langs_arg, ":;,");
ce426f
+                  if (lang == NULL)
ce426f
+                    break;
ce426f
+                  install_langs_count++;
ce426f
+                  install_langs_arg = NULL;
ce426f
+                }
ce426f
+	      free (ila_start);
ce426f
+	      /* Copy the list.  */
ce426f
+	      install_langs_list = (char **)xmalloc (sizeof(char *) * install_langs_count);
ce426f
+	      install_langs_arg = ila_start = strdup (optarg);
ce426f
+	      install_langs_count = 0;
ce426f
+	      while (true)
ce426f
+                {
ce426f
+                  lang = strtok(install_langs_arg, ":;,");
ce426f
+                  if (lang == NULL)
ce426f
+                    break;
ce426f
+                  install_langs_list[install_langs_count] = lang;
ce426f
+		  install_langs_count++;
ce426f
+                  install_langs_arg = NULL;
ce426f
+                }
ce426f
+            }
ce426f
+          break;
ce426f
+
ce426f
+        case '?':
ce426f
+          /* getopt_long already printed an error message. */
ce426f
+          usage ();
ce426f
+          exit (0);
ce426f
+
ce426f
+        default:
ce426f
+          abort ();
ce426f
+        }
ce426f
+    }
ce426f
+  tmpl_ah.fname = NULL;
ce426f
+  if (optind < argc)
ce426f
+    tmpl_ah.fname = argv[optind];
ce426f
+  if (optind + 1 < argc)
ce426f
+    new_locar_fname = argv[optind + 1];
ce426f
+  if (verbose)
ce426f
+    {
ce426f
+      if (tmpl_ah.fname)
ce426f
+        printf("input archive file specified on command line: %s\n",
ce426f
+               tmpl_ah.fname);
ce426f
+      else
ce426f
+        printf("using default input archive file.\n");
ce426f
+      if (new_locar_fname)
ce426f
+        printf("output archive file specified on command line: %s\n",
ce426f
+               new_locar_fname);
ce426f
+      else
ce426f
+        printf("using default output archive file.\n");
ce426f
+    }
ce426f
+
ce426f
   dirp = opendir (loc_path);
ce426f
   if (dirp == NULL)
ce426f
     error (EXIT_FAILURE, errno, "cannot open directory \"%s\"", loc_path);
ce426f
 
ce426f
   open_tmpl_archive (&tmpl_ah);
ce426f
 
ce426f
-  unlink (locar_file);
ce426f
+  if (new_locar_fname)
ce426f
+    unlink (new_locar_fname);
ce426f
+  else
ce426f
+    unlink (locar_file);
ce426f
+
ce426f
   primary = getenv ("LC_ALL");
ce426f
   if (primary == NULL)
ce426f
     primary = getenv ("LANG");
ce426f
@@ -560,7 +770,8 @@ int main ()
ce426f
 	  && strncmp (primary, "zh", 2) != 0)
ce426f
 	{
ce426f
 	  char *ptr = malloc (strlen (primary) + strlen (".utf8") + 1), *p, *q;
ce426f
-
ce426f
+	  /* This leads to invalid locales sometimes:
ce426f
+	     de_DE.iso885915@euro -> de_DE.utf8@euro */
ce426f
 	  if (ptr != NULL)
ce426f
 	    {
ce426f
 	      p = ptr;
ce426f
@@ -623,10 +834,19 @@ int main ()
ce426f
       cnt++;
ce426f
     }
ce426f
   closedir (dirp);
ce426f
-  fill_archive (&tmpl_ah, cnt, list, primary);
ce426f
+  /* Store the archive to the file specified as the second argument on the
ce426f
+     command line or the default locale archive.  */
ce426f
+  fill_archive (&tmpl_ah, new_locar_fname,
ce426f
+                install_langs_count, install_langs_list,
ce426f
+                cnt, list, primary);
ce426f
   close_archive (&tmpl_ah);
ce426f
   truncate (tmpl_file, 0);
ce426f
-  char *argv[] = { "/usr/sbin/tzdata-update", NULL };
ce426f
-  execve (argv[0], (char *const *)argv, (char *const *)&argv[1]);
ce426f
+  if (install_langs_count > 0)
ce426f
+    {
ce426f
+      free (ila_start);
ce426f
+      free (install_langs_list);
ce426f
+    }
ce426f
+  char *tz_argv[] = { "/usr/sbin/tzdata-update", NULL };
ce426f
+  execve (tz_argv[0], (char *const *)tz_argv, (char *const *)&tz_argv[1]);
ce426f
   exit (0);
ce426f
 }