1 | /* Load needed message catalogs
|
---|
2 | Copyright (C) 1995, 1996, 1997 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 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 = 0;
|
---|
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 | }
|
---|