~ubuntu-branches/ubuntu/trusty/librep/trusty

« back to all changes in this revision

Viewing changes to intl/loadmsgcat.c

  • Committer: Bazaar Package Importer
  • Author(s): Christian Marillat
  • Date: 2001-11-13 15:06:22 UTC
  • Revision ID: james.westby@ubuntu.com-20011113150622-vgmgmk6srj3kldr3
Tags: upstream-0.15.2
ImportĀ upstreamĀ versionĀ 0.15.2

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/* Load needed message catalogs.
 
2
   Copyright (C) 1995, 1996, 1997, 1998 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 Foundation,
 
16
   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 ISO C functions.  This is required by the standard
 
45
   because some ISO C functions will require linking with this object
 
46
   file and the name space must not be polluted.  */
 
47
# define open   __open
 
48
# define close  __close
 
49
# define read   __read
 
50
# define mmap   __mmap
 
51
# define munmap __munmap
 
52
#endif
 
53
 
 
54
/* We need a sign, whether a new catalog was loaded, which can be associated
 
55
   with all translations.  This is important if the translations are
 
56
   cached by one of GCC's features.  */
 
57
int _nl_msg_cat_cntr = 0;
 
58
 
 
59
 
 
60
/* Load the message catalogs specified by FILENAME.  If it is no valid
 
61
   message catalog do nothing.  */
 
62
void
 
63
internal_function
 
64
_nl_load_domain (domain_file)
 
65
     struct loaded_l10nfile *domain_file;
 
66
{
 
67
  int fd;
 
68
  size_t size;
 
69
  struct stat st;
 
70
  struct mo_file_header *data = (struct mo_file_header *) -1;
 
71
#if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
 
72
    || defined _LIBC
 
73
  int use_mmap = 0;
 
74
#endif
 
75
  struct loaded_domain *domain;
 
76
 
 
77
  domain_file->decided = 1;
 
78
  domain_file->data = NULL;
 
79
 
 
80
  /* If the record does not represent a valid locale the FILENAME
 
81
     might be NULL.  This can happen when according to the given
 
82
     specification the locale file name is different for XPG and CEN
 
83
     syntax.  */
 
84
  if (domain_file->filename == NULL)
 
85
    return;
 
86
 
 
87
  /* Try to open the addressed file.  */
 
88
  fd = open (domain_file->filename, O_RDONLY);
 
89
  if (fd == -1)
 
90
    return;
 
91
 
 
92
  /* We must know about the size of the file.  */
 
93
  if (fstat (fd, &st) != 0
 
94
      || (size = (size_t) st.st_size) != st.st_size
 
95
      || size < sizeof (struct mo_file_header))
 
96
    {
 
97
      /* Something went wrong.  */
 
98
      close (fd);
 
99
      return;
 
100
    }
 
101
 
 
102
#if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
 
103
    || defined _LIBC
 
104
  /* Now we are ready to load the file.  If mmap() is available we try
 
105
     this first.  If not available or it failed we try to load it.  */
 
106
  data = (struct mo_file_header *) mmap (NULL, size, PROT_READ,
 
107
                                         MAP_PRIVATE, fd, 0);
 
108
 
 
109
  if (data != (struct mo_file_header *) -1)
 
110
    {
 
111
      /* mmap() call was successful.  */
 
112
      close (fd);
 
113
      use_mmap = 1;
 
114
    }
 
115
#endif
 
116
 
 
117
  /* If the data is not yet available (i.e. mmap'ed) we try to load
 
118
     it manually.  */
 
119
  if (data == (struct mo_file_header *) -1)
 
120
    {
 
121
      size_t to_read;
 
122
      char *read_ptr;
 
123
 
 
124
      data = (struct mo_file_header *) malloc (size);
 
125
      if (data == NULL)
 
126
        return;
 
127
 
 
128
      to_read = size;
 
129
      read_ptr = (char *) data;
 
130
      do
 
131
        {
 
132
          long int nb = (long int) read (fd, read_ptr, to_read);
 
133
          if (nb == -1)
 
134
            {
 
135
              close (fd);
 
136
              return;
 
137
            }
 
138
 
 
139
          read_ptr += nb;
 
140
          to_read -= nb;
 
141
        }
 
142
      while (to_read > 0);
 
143
 
 
144
      close (fd);
 
145
    }
 
146
 
 
147
  /* Using the magic number we can test whether it really is a message
 
148
     catalog file.  */
 
149
  if (data->magic != _MAGIC && data->magic != _MAGIC_SWAPPED)
 
150
    {
 
151
      /* The magic number is wrong: not a message catalog file.  */
 
152
#if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
 
153
    || defined _LIBC
 
154
      if (use_mmap)
 
155
        munmap ((caddr_t) data, size);
 
156
      else
 
157
#endif
 
158
        free (data);
 
159
      return;
 
160
    }
 
161
 
 
162
  domain_file->data
 
163
    = (struct loaded_domain *) malloc (sizeof (struct loaded_domain));
 
164
  if (domain_file->data == NULL)
 
165
    return;
 
166
 
 
167
  domain = (struct loaded_domain *) domain_file->data;
 
168
  domain->data = (char *) data;
 
169
#if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
 
170
    || defined _LIBC
 
171
  domain->use_mmap = use_mmap;
 
172
#endif
 
173
  domain->mmap_size = size;
 
174
  domain->must_swap = data->magic != _MAGIC;
 
175
 
 
176
  /* Fill in the information about the available tables.  */
 
177
  switch (W (domain->must_swap, data->revision))
 
178
    {
 
179
    case 0:
 
180
      domain->nstrings = W (domain->must_swap, data->nstrings);
 
181
      domain->orig_tab = (struct string_desc *)
 
182
        ((char *) data + W (domain->must_swap, data->orig_tab_offset));
 
183
      domain->trans_tab = (struct string_desc *)
 
184
        ((char *) data + W (domain->must_swap, data->trans_tab_offset));
 
185
      domain->hash_size = W (domain->must_swap, data->hash_tab_size);
 
186
      domain->hash_tab = (nls_uint32 *)
 
187
        ((char *) data + W (domain->must_swap, data->hash_tab_offset));
 
188
      break;
 
189
    default:
 
190
      /* This is an illegal revision.  */
 
191
#if (defined HAVE_MMAP && defined HAVE_MUNMAP && !defined DISALLOW_MMAP) \
 
192
    || defined _LIBC
 
193
      if (use_mmap)
 
194
        munmap ((caddr_t) data, size);
 
195
      else
 
196
#endif
 
197
        free (data);
 
198
      free (domain);
 
199
      domain_file->data = NULL;
 
200
      return;
 
201
    }
 
202
 
 
203
  /* Show that one domain is changed.  This might make some cached
 
204
     translations invalid.  */
 
205
  ++_nl_msg_cat_cntr;
 
206
}
 
207
 
 
208
 
 
209
#ifdef _LIBC
 
210
void
 
211
internal_function
 
212
_nl_unload_domain (domain)
 
213
     struct loaded_domain *domain;
 
214
{
 
215
  if (domain->use_mmap)
 
216
    munmap ((caddr_t) domain->data, domain->mmap_size);
 
217
  else
 
218
    free ((void *) domain->data);
 
219
 
 
220
  free (domain);
 
221
}
 
222
#endif