home *** CD-ROM | disk | FTP | other *** search
/ Geek Gadgets 1 / ADE-1.bin / ade-dist / textutils-1.19-src.tgz / tar.out / fsf / textutils / intl / loadmsgcat.c < prev    next >
C/C++ Source or Header  |  1996-09-28  |  6KB  |  200 lines

  1. /* loadmsgcat.c -- load needed message catalogs
  2.    Copyright (C) 1995, 1996 Free Software Foundation, Inc.
  3.  
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation; either version 2, or (at your option)
  7. any later version.
  8.  
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12. GNU General Public License for more details.
  13.  
  14. You should have received a copy of the GNU General Public License
  15. along with this program; if not, write to the Free Software
  16. Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
  17.  
  18. #ifdef HAVE_CONFIG_H
  19. # include <config.h>
  20. #endif
  21.  
  22. #include <fcntl.h>
  23. #include <sys/types.h>
  24. #include <sys/stat.h>
  25.  
  26. #if defined STDC_HEADERS || defined _LIBC
  27. # include <stdlib.h>
  28. #endif
  29.  
  30. #if defined HAVE_UNISTD_H || defined _LIBC
  31. # include <unistd.h>
  32. #endif
  33.  
  34. #if (defined HAVE_MMAP && defined HAVE_MUNMAP) || defined _LIBC
  35. # include <sys/mman.h>
  36. #endif
  37.  
  38. #include "gettext.h"
  39. #include "gettextP.h"
  40.  
  41. /* @@ end of prolog @@ */
  42.  
  43. #ifdef _LIBC
  44. /* Rename the non ANSI C functions.  This is required by the standard
  45.    because some ANSI C functions will require linking with this object
  46.    file and the name space must not be polluted.  */
  47. # define fstat  __fstat
  48. # define open   __open
  49. # define close  __close
  50. # define read   __read
  51. # define mmap   __mmap
  52. # define munmap __munmap
  53. #endif
  54.  
  55. /* We need a sign, whether a new catalog was loaded, which can be associated
  56.    with all translations.  This is important if the translations are
  57.    cached by one of GCC's features.  */
  58. int _nl_msg_cat_cntr;
  59.  
  60.  
  61. /* Load the message catalogs specified by FILENAME.  If it is no valid
  62.    message catalog do nothing.  */
  63. void
  64. _nl_load_domain (domain_file)
  65.      struct loaded_l10nfile *domain_file;
  66. {
  67.   int fd;
  68.   struct stat st;
  69.   struct mo_file_header *data = (struct mo_file_header *) -1;
  70. #if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
  71.     || defined _LIBC
  72.   int use_mmap = 0;
  73. #endif
  74.   struct loaded_domain *domain;
  75.  
  76.   domain_file->decided = 1;
  77.   domain_file->data = NULL;
  78.  
  79.   /* If the record does not represent a valid locale the FILENAME
  80.      might be NULL.  This can happen when according to the given
  81.      specification the locale file name is different for XPG and CEN
  82.      syntax.  */
  83.   if (domain_file->filename == NULL)
  84.     return;
  85.  
  86.   /* Try to open the addressed file.  */
  87.   fd = open (domain_file->filename, O_RDONLY);
  88.   if (fd == -1)
  89.     return;
  90.  
  91.   /* We must know about the size of the file.  */
  92.   if (fstat (fd, &st) != 0
  93.       && st.st_size < (off_t) sizeof (struct mo_file_header))
  94.     {
  95.       /* Something went wrong.  */
  96.       close (fd);
  97.       return;
  98.     }
  99.  
  100. #if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
  101.     || defined _LIBC
  102.   /* Now we are ready to load the file.  If mmap() is available we try
  103.      this first.  If not available or it failed we try to load it.  */
  104.   data = (struct mo_file_header *) mmap (NULL, st.st_size, PROT_READ,
  105.                      MAP_PRIVATE, fd, 0);
  106.  
  107.   if (data != (struct mo_file_header *) -1)
  108.     {
  109.       /* mmap() call was successful.  */
  110.       close (fd);
  111.       use_mmap = 1;
  112.     }
  113. #endif
  114.  
  115.   /* If the data is not yet available (i.e. mmap'ed) we try to load
  116.      it manually.  */
  117.   if (data == (struct mo_file_header *) -1)
  118.     {
  119.       off_t to_read;
  120.       char *read_ptr;
  121.  
  122.       data = (struct mo_file_header *) malloc (st.st_size);
  123.       if (data == NULL)
  124.     return;
  125.  
  126.       to_read = st.st_size;
  127.       read_ptr = (char *) data;
  128.       do
  129.     {
  130.       long int nb = (long int) read (fd, read_ptr, to_read);
  131.       if (nb == -1)
  132.         {
  133.           close (fd);
  134.           return;
  135.         }
  136.  
  137.       read_ptr += nb;
  138.       to_read -= nb;
  139.     }
  140.       while (to_read > 0);
  141.  
  142.       close (fd);
  143.     }
  144.  
  145.   /* Using the magic number we can test whether it really is a message
  146.      catalog file.  */
  147.   if (data->magic != _MAGIC && data->magic != _MAGIC_SWAPPED)
  148.     {
  149.       /* The magic number is wrong: not a message catalog file.  */
  150. #if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
  151.     || defined _LIBC
  152.       if (use_mmap)
  153.     munmap ((caddr_t) data, st.st_size);
  154.       else
  155. #endif
  156.     free (data);
  157.       return;
  158.     }
  159.  
  160.   domain_file->data
  161.     = (struct loaded_domain *) malloc (sizeof (struct loaded_domain));
  162.   if (domain_file->data == NULL)
  163.     return;
  164.  
  165.   domain = (struct loaded_domain *) domain_file->data;
  166.   domain->data = (char *) data;
  167.   domain->must_swap = data->magic != _MAGIC;
  168.  
  169.   /* Fill in the information about the available tables.  */
  170.   switch (W (domain->must_swap, data->revision))
  171.     {
  172.     case 0:
  173.       domain->nstrings = W (domain->must_swap, data->nstrings);
  174.       domain->orig_tab = (struct string_desc *)
  175.     ((char *) data + W (domain->must_swap, data->orig_tab_offset));
  176.       domain->trans_tab = (struct string_desc *)
  177.     ((char *) data + W (domain->must_swap, data->trans_tab_offset));
  178.       domain->hash_size = W (domain->must_swap, data->hash_tab_size);
  179.       domain->hash_tab = (nls_uint32 *)
  180.     ((char *) data + W (domain->must_swap, data->hash_tab_offset));
  181.       break;
  182.     default:
  183.       /* This is an illegal revision.  */
  184. #if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
  185.     || defined _LIBC
  186.       if (use_mmap)
  187.     munmap ((caddr_t) data, st.st_size);
  188.       else
  189. #endif
  190.     free (data);
  191.       free (domain);
  192.       domain_file->data = NULL;
  193.       return;
  194.     }
  195.  
  196.   /* Show that one domain is changed.  This might make some cached
  197.      translations invalid.  */
  198.   ++_nl_msg_cat_cntr;
  199. }
  200.