1*3d8817e4Smiod /* resres.c: read_res_file and write_res_file implementation for windres.
2*3d8817e4Smiod Copyright 1998, 1999, 2001, 2002 Free Software Foundation, Inc.
3*3d8817e4Smiod Written by Anders Norlander <anorland@hem2.passagen.se>.
4*3d8817e4Smiod
5*3d8817e4Smiod This file is part of GNU Binutils.
6*3d8817e4Smiod
7*3d8817e4Smiod This program is free software; you can redistribute it and/or modify
8*3d8817e4Smiod it under the terms of the GNU General Public License as published by
9*3d8817e4Smiod the Free Software Foundation; either version 2 of the License, or
10*3d8817e4Smiod (at your option) any later version.
11*3d8817e4Smiod
12*3d8817e4Smiod This program is distributed in the hope that it will be useful,
13*3d8817e4Smiod but WITHOUT ANY WARRANTY; without even the implied warranty of
14*3d8817e4Smiod MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15*3d8817e4Smiod GNU General Public License for more details.
16*3d8817e4Smiod
17*3d8817e4Smiod You should have received a copy of the GNU General Public License
18*3d8817e4Smiod along with this program; if not, write to the Free Software
19*3d8817e4Smiod Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
20*3d8817e4Smiod 02110-1301, USA. */
21*3d8817e4Smiod
22*3d8817e4Smiod /* FIXME: This file does not work correctly in a cross configuration.
23*3d8817e4Smiod It assumes that it can use fread and fwrite to read and write
24*3d8817e4Smiod integers. It does no swapping. */
25*3d8817e4Smiod
26*3d8817e4Smiod #include "bfd.h"
27*3d8817e4Smiod #include "bucomm.h"
28*3d8817e4Smiod #include "libiberty.h"
29*3d8817e4Smiod #include "windres.h"
30*3d8817e4Smiod
31*3d8817e4Smiod #include <assert.h>
32*3d8817e4Smiod #include <time.h>
33*3d8817e4Smiod
34*3d8817e4Smiod struct res_hdr
35*3d8817e4Smiod {
36*3d8817e4Smiod unsigned long data_size;
37*3d8817e4Smiod unsigned long header_size;
38*3d8817e4Smiod };
39*3d8817e4Smiod
40*3d8817e4Smiod static void write_res_directory
41*3d8817e4Smiod PARAMS ((const struct res_directory *,
42*3d8817e4Smiod const struct res_id *, const struct res_id *,
43*3d8817e4Smiod int *, int));
44*3d8817e4Smiod static void write_res_resource
45*3d8817e4Smiod PARAMS ((const struct res_id *, const struct res_id *,
46*3d8817e4Smiod const struct res_resource *, int *));
47*3d8817e4Smiod static void write_res_bin
48*3d8817e4Smiod PARAMS ((const struct res_resource *, const struct res_id *,
49*3d8817e4Smiod const struct res_id *, const struct res_res_info *));
50*3d8817e4Smiod
51*3d8817e4Smiod static void write_res_id PARAMS ((const struct res_id *));
52*3d8817e4Smiod static void write_res_info PARAMS ((const struct res_res_info *));
53*3d8817e4Smiod static void write_res_data PARAMS ((const void *, size_t, int));
54*3d8817e4Smiod static void write_res_header
55*3d8817e4Smiod PARAMS ((unsigned long, const struct res_id *, const struct res_id *,
56*3d8817e4Smiod const struct res_res_info *));
57*3d8817e4Smiod
58*3d8817e4Smiod static int read_resource_entry PARAMS ((void));
59*3d8817e4Smiod static void read_res_data PARAMS ((void *, size_t, int));
60*3d8817e4Smiod static void read_res_id PARAMS ((struct res_id *));
61*3d8817e4Smiod static unichar *read_unistring PARAMS ((int *));
62*3d8817e4Smiod static void skip_null_resource PARAMS ((void));
63*3d8817e4Smiod
64*3d8817e4Smiod static unsigned long get_id_size PARAMS ((const struct res_id *));
65*3d8817e4Smiod static void res_align_file PARAMS ((void));
66*3d8817e4Smiod
67*3d8817e4Smiod static void
68*3d8817e4Smiod res_add_resource
69*3d8817e4Smiod PARAMS ((struct res_resource *, const struct res_id *,
70*3d8817e4Smiod const struct res_id *, int, int));
71*3d8817e4Smiod
72*3d8817e4Smiod void
73*3d8817e4Smiod res_append_resource
74*3d8817e4Smiod PARAMS ((struct res_directory **, struct res_resource *,
75*3d8817e4Smiod int, const struct res_id *, int));
76*3d8817e4Smiod
77*3d8817e4Smiod static struct res_directory *resources = NULL;
78*3d8817e4Smiod
79*3d8817e4Smiod static FILE *fres;
80*3d8817e4Smiod static const char *filename;
81*3d8817e4Smiod
82*3d8817e4Smiod extern char *program_name;
83*3d8817e4Smiod
84*3d8817e4Smiod /* Read resource file */
85*3d8817e4Smiod struct res_directory *
read_res_file(fn)86*3d8817e4Smiod read_res_file (fn)
87*3d8817e4Smiod const char *fn;
88*3d8817e4Smiod {
89*3d8817e4Smiod filename = fn;
90*3d8817e4Smiod fres = fopen (filename, "rb");
91*3d8817e4Smiod if (fres == NULL)
92*3d8817e4Smiod fatal ("can't open `%s' for output: %s", filename, strerror (errno));
93*3d8817e4Smiod
94*3d8817e4Smiod skip_null_resource ();
95*3d8817e4Smiod
96*3d8817e4Smiod while (read_resource_entry ())
97*3d8817e4Smiod ;
98*3d8817e4Smiod
99*3d8817e4Smiod fclose (fres);
100*3d8817e4Smiod
101*3d8817e4Smiod return resources;
102*3d8817e4Smiod }
103*3d8817e4Smiod
104*3d8817e4Smiod /* Write resource file */
105*3d8817e4Smiod void
write_res_file(fn,resdir)106*3d8817e4Smiod write_res_file (fn, resdir)
107*3d8817e4Smiod const char *fn;
108*3d8817e4Smiod const struct res_directory *resdir;
109*3d8817e4Smiod {
110*3d8817e4Smiod int language;
111*3d8817e4Smiod static const unsigned char sign[] =
112*3d8817e4Smiod {0x00, 0x00, 0x00, 0x00, 0x20, 0x00, 0x00, 0x00,
113*3d8817e4Smiod 0xFF, 0xFF, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0x00,
114*3d8817e4Smiod 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
115*3d8817e4Smiod 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
116*3d8817e4Smiod long fpos;
117*3d8817e4Smiod
118*3d8817e4Smiod filename = fn;
119*3d8817e4Smiod
120*3d8817e4Smiod fres = fopen (filename, "wb");
121*3d8817e4Smiod if (fres == NULL)
122*3d8817e4Smiod fatal ("can't open `%s' for output: %s", filename, strerror (errno));
123*3d8817e4Smiod
124*3d8817e4Smiod /* Write 32 bit resource signature */
125*3d8817e4Smiod write_res_data (sign, sizeof (sign), 1);
126*3d8817e4Smiod
127*3d8817e4Smiod /* write resources */
128*3d8817e4Smiod
129*3d8817e4Smiod language = -1;
130*3d8817e4Smiod write_res_directory (resdir, (const struct res_id *) NULL,
131*3d8817e4Smiod (const struct res_id *) NULL, &language, 1);
132*3d8817e4Smiod
133*3d8817e4Smiod /* end file on DWORD boundary */
134*3d8817e4Smiod fpos = ftell (fres);
135*3d8817e4Smiod if (fpos % 4)
136*3d8817e4Smiod write_res_data (sign, fpos % 4, 1);
137*3d8817e4Smiod
138*3d8817e4Smiod fclose (fres);
139*3d8817e4Smiod }
140*3d8817e4Smiod
141*3d8817e4Smiod /* Read a resource entry, returns 0 when all resources are read */
142*3d8817e4Smiod static int
read_resource_entry(void)143*3d8817e4Smiod read_resource_entry (void)
144*3d8817e4Smiod {
145*3d8817e4Smiod struct res_id type;
146*3d8817e4Smiod struct res_id name;
147*3d8817e4Smiod struct res_res_info resinfo;
148*3d8817e4Smiod struct res_hdr reshdr;
149*3d8817e4Smiod long version;
150*3d8817e4Smiod void *buff;
151*3d8817e4Smiod
152*3d8817e4Smiod struct res_resource *r;
153*3d8817e4Smiod
154*3d8817e4Smiod res_align_file ();
155*3d8817e4Smiod
156*3d8817e4Smiod /* Read header */
157*3d8817e4Smiod if (fread (&reshdr, sizeof (reshdr), 1, fres) != 1)
158*3d8817e4Smiod return 0;
159*3d8817e4Smiod
160*3d8817e4Smiod /* read resource type */
161*3d8817e4Smiod read_res_id (&type);
162*3d8817e4Smiod /* read resource id */
163*3d8817e4Smiod read_res_id (&name);
164*3d8817e4Smiod
165*3d8817e4Smiod res_align_file ();
166*3d8817e4Smiod
167*3d8817e4Smiod /* Read additional resource header */
168*3d8817e4Smiod read_res_data (&resinfo.version, sizeof (resinfo.version), 1);
169*3d8817e4Smiod read_res_data (&resinfo.memflags, sizeof (resinfo.memflags), 1);
170*3d8817e4Smiod read_res_data (&resinfo.language, sizeof (resinfo.language), 1);
171*3d8817e4Smiod read_res_data (&version, sizeof (version), 1);
172*3d8817e4Smiod read_res_data (&resinfo.characteristics, sizeof (resinfo.characteristics), 1);
173*3d8817e4Smiod
174*3d8817e4Smiod res_align_file ();
175*3d8817e4Smiod
176*3d8817e4Smiod /* Allocate buffer for data */
177*3d8817e4Smiod buff = res_alloc (reshdr.data_size);
178*3d8817e4Smiod /* Read data */
179*3d8817e4Smiod read_res_data (buff, reshdr.data_size, 1);
180*3d8817e4Smiod /* Convert binary data to resource */
181*3d8817e4Smiod r = bin_to_res (type, buff, reshdr.data_size, 0);
182*3d8817e4Smiod r->res_info = resinfo;
183*3d8817e4Smiod /* Add resource to resource directory */
184*3d8817e4Smiod res_add_resource (r, &type, &name, resinfo.language, 0);
185*3d8817e4Smiod
186*3d8817e4Smiod return 1;
187*3d8817e4Smiod }
188*3d8817e4Smiod
189*3d8817e4Smiod /* write resource directory to binary resource file */
190*3d8817e4Smiod static void
write_res_directory(rd,type,name,language,level)191*3d8817e4Smiod write_res_directory (rd, type, name, language, level)
192*3d8817e4Smiod const struct res_directory *rd;
193*3d8817e4Smiod const struct res_id *type;
194*3d8817e4Smiod const struct res_id *name;
195*3d8817e4Smiod int *language;
196*3d8817e4Smiod int level;
197*3d8817e4Smiod {
198*3d8817e4Smiod const struct res_entry *re;
199*3d8817e4Smiod
200*3d8817e4Smiod for (re = rd->entries; re != NULL; re = re->next)
201*3d8817e4Smiod {
202*3d8817e4Smiod switch (level)
203*3d8817e4Smiod {
204*3d8817e4Smiod case 1:
205*3d8817e4Smiod /* If we're at level 1, the key of this resource is the
206*3d8817e4Smiod type. This normally duplicates the information we have
207*3d8817e4Smiod stored with the resource itself, but we need to remember
208*3d8817e4Smiod the type if this is a user define resource type. */
209*3d8817e4Smiod type = &re->id;
210*3d8817e4Smiod break;
211*3d8817e4Smiod
212*3d8817e4Smiod case 2:
213*3d8817e4Smiod /* If we're at level 2, the key of this resource is the name
214*3d8817e4Smiod we are going to use in the rc printout. */
215*3d8817e4Smiod name = &re->id;
216*3d8817e4Smiod break;
217*3d8817e4Smiod
218*3d8817e4Smiod case 3:
219*3d8817e4Smiod /* If we're at level 3, then this key represents a language.
220*3d8817e4Smiod Use it to update the current language. */
221*3d8817e4Smiod if (!re->id.named
222*3d8817e4Smiod && re->id.u.id != (unsigned long) *language
223*3d8817e4Smiod && (re->id.u.id & 0xffff) == re->id.u.id)
224*3d8817e4Smiod {
225*3d8817e4Smiod *language = re->id.u.id;
226*3d8817e4Smiod }
227*3d8817e4Smiod break;
228*3d8817e4Smiod
229*3d8817e4Smiod default:
230*3d8817e4Smiod break;
231*3d8817e4Smiod }
232*3d8817e4Smiod
233*3d8817e4Smiod if (re->subdir)
234*3d8817e4Smiod write_res_directory (re->u.dir, type, name, language, level + 1);
235*3d8817e4Smiod else
236*3d8817e4Smiod {
237*3d8817e4Smiod if (level == 3)
238*3d8817e4Smiod {
239*3d8817e4Smiod /* This is the normal case: the three levels are
240*3d8817e4Smiod TYPE/NAME/LANGUAGE. NAME will have been set at level
241*3d8817e4Smiod 2, and represents the name to use. We probably just
242*3d8817e4Smiod set LANGUAGE, and it will probably match what the
243*3d8817e4Smiod resource itself records if anything. */
244*3d8817e4Smiod write_res_resource (type, name, re->u.res, language);
245*3d8817e4Smiod }
246*3d8817e4Smiod else
247*3d8817e4Smiod {
248*3d8817e4Smiod fprintf (stderr, "// Resource at unexpected level %d\n", level);
249*3d8817e4Smiod write_res_resource (type, (struct res_id *) NULL, re->u.res,
250*3d8817e4Smiod language);
251*3d8817e4Smiod }
252*3d8817e4Smiod }
253*3d8817e4Smiod }
254*3d8817e4Smiod
255*3d8817e4Smiod }
256*3d8817e4Smiod
257*3d8817e4Smiod static void
write_res_resource(type,name,res,language)258*3d8817e4Smiod write_res_resource (type, name, res, language)
259*3d8817e4Smiod const struct res_id *type;
260*3d8817e4Smiod const struct res_id *name;
261*3d8817e4Smiod const struct res_resource *res;
262*3d8817e4Smiod int *language ATTRIBUTE_UNUSED;
263*3d8817e4Smiod {
264*3d8817e4Smiod int rt;
265*3d8817e4Smiod
266*3d8817e4Smiod switch (res->type)
267*3d8817e4Smiod {
268*3d8817e4Smiod default:
269*3d8817e4Smiod abort ();
270*3d8817e4Smiod
271*3d8817e4Smiod case RES_TYPE_ACCELERATOR:
272*3d8817e4Smiod rt = RT_ACCELERATOR;
273*3d8817e4Smiod break;
274*3d8817e4Smiod
275*3d8817e4Smiod case RES_TYPE_BITMAP:
276*3d8817e4Smiod rt = RT_BITMAP;
277*3d8817e4Smiod break;
278*3d8817e4Smiod
279*3d8817e4Smiod case RES_TYPE_CURSOR:
280*3d8817e4Smiod rt = RT_CURSOR;
281*3d8817e4Smiod break;
282*3d8817e4Smiod
283*3d8817e4Smiod case RES_TYPE_GROUP_CURSOR:
284*3d8817e4Smiod rt = RT_GROUP_CURSOR;
285*3d8817e4Smiod break;
286*3d8817e4Smiod
287*3d8817e4Smiod case RES_TYPE_DIALOG:
288*3d8817e4Smiod rt = RT_DIALOG;
289*3d8817e4Smiod break;
290*3d8817e4Smiod
291*3d8817e4Smiod case RES_TYPE_FONT:
292*3d8817e4Smiod rt = RT_FONT;
293*3d8817e4Smiod break;
294*3d8817e4Smiod
295*3d8817e4Smiod case RES_TYPE_FONTDIR:
296*3d8817e4Smiod rt = RT_FONTDIR;
297*3d8817e4Smiod break;
298*3d8817e4Smiod
299*3d8817e4Smiod case RES_TYPE_ICON:
300*3d8817e4Smiod rt = RT_ICON;
301*3d8817e4Smiod break;
302*3d8817e4Smiod
303*3d8817e4Smiod case RES_TYPE_GROUP_ICON:
304*3d8817e4Smiod rt = RT_GROUP_ICON;
305*3d8817e4Smiod break;
306*3d8817e4Smiod
307*3d8817e4Smiod case RES_TYPE_MENU:
308*3d8817e4Smiod rt = RT_MENU;
309*3d8817e4Smiod break;
310*3d8817e4Smiod
311*3d8817e4Smiod case RES_TYPE_MESSAGETABLE:
312*3d8817e4Smiod rt = RT_MESSAGETABLE;
313*3d8817e4Smiod break;
314*3d8817e4Smiod
315*3d8817e4Smiod case RES_TYPE_RCDATA:
316*3d8817e4Smiod rt = RT_RCDATA;
317*3d8817e4Smiod break;
318*3d8817e4Smiod
319*3d8817e4Smiod case RES_TYPE_STRINGTABLE:
320*3d8817e4Smiod rt = RT_STRING;
321*3d8817e4Smiod break;
322*3d8817e4Smiod
323*3d8817e4Smiod case RES_TYPE_USERDATA:
324*3d8817e4Smiod rt = 0;
325*3d8817e4Smiod break;
326*3d8817e4Smiod
327*3d8817e4Smiod case RES_TYPE_VERSIONINFO:
328*3d8817e4Smiod rt = RT_VERSION;
329*3d8817e4Smiod break;
330*3d8817e4Smiod }
331*3d8817e4Smiod
332*3d8817e4Smiod if (rt != 0
333*3d8817e4Smiod && type != NULL
334*3d8817e4Smiod && (type->named || type->u.id != (unsigned long) rt))
335*3d8817e4Smiod {
336*3d8817e4Smiod fprintf (stderr, "// Unexpected resource type mismatch: ");
337*3d8817e4Smiod res_id_print (stderr, *type, 1);
338*3d8817e4Smiod fprintf (stderr, " != %d", rt);
339*3d8817e4Smiod abort ();
340*3d8817e4Smiod }
341*3d8817e4Smiod
342*3d8817e4Smiod write_res_bin (res, type, name, &res->res_info);
343*3d8817e4Smiod return;
344*3d8817e4Smiod }
345*3d8817e4Smiod
346*3d8817e4Smiod /* Write a resource in binary resource format */
347*3d8817e4Smiod static void
write_res_bin(res,type,name,resinfo)348*3d8817e4Smiod write_res_bin (res, type, name, resinfo)
349*3d8817e4Smiod const struct res_resource *res;
350*3d8817e4Smiod const struct res_id *type;
351*3d8817e4Smiod const struct res_id *name;
352*3d8817e4Smiod const struct res_res_info *resinfo;
353*3d8817e4Smiod {
354*3d8817e4Smiod unsigned long datasize = 0;
355*3d8817e4Smiod const struct bindata *bin_rep, *data;
356*3d8817e4Smiod
357*3d8817e4Smiod bin_rep = res_to_bin (res, 0);
358*3d8817e4Smiod for (data = bin_rep; data != NULL; data = data->next)
359*3d8817e4Smiod datasize += data->length;
360*3d8817e4Smiod
361*3d8817e4Smiod write_res_header (datasize, type, name, resinfo);
362*3d8817e4Smiod
363*3d8817e4Smiod for (data = bin_rep; data != NULL; data = data->next)
364*3d8817e4Smiod write_res_data (data->data, data->length, 1);
365*3d8817e4Smiod }
366*3d8817e4Smiod
367*3d8817e4Smiod /* Get number of bytes needed to store an id in binary format */
368*3d8817e4Smiod static unsigned long
get_id_size(id)369*3d8817e4Smiod get_id_size (id)
370*3d8817e4Smiod const struct res_id *id;
371*3d8817e4Smiod {
372*3d8817e4Smiod if (id->named)
373*3d8817e4Smiod return sizeof (unichar) * (id->u.n.length + 1);
374*3d8817e4Smiod else
375*3d8817e4Smiod return sizeof (unichar) * 2;
376*3d8817e4Smiod }
377*3d8817e4Smiod
378*3d8817e4Smiod /* Write a resource header */
379*3d8817e4Smiod static void
write_res_header(datasize,type,name,resinfo)380*3d8817e4Smiod write_res_header (datasize, type, name, resinfo)
381*3d8817e4Smiod unsigned long datasize;
382*3d8817e4Smiod const struct res_id *type;
383*3d8817e4Smiod const struct res_id *name;
384*3d8817e4Smiod const struct res_res_info *resinfo;
385*3d8817e4Smiod {
386*3d8817e4Smiod struct res_hdr reshdr;
387*3d8817e4Smiod reshdr.data_size = datasize;
388*3d8817e4Smiod reshdr.header_size = 24 + get_id_size (type) + get_id_size (name);
389*3d8817e4Smiod
390*3d8817e4Smiod reshdr.header_size = (reshdr.header_size + 3) & ~3;
391*3d8817e4Smiod
392*3d8817e4Smiod res_align_file ();
393*3d8817e4Smiod write_res_data (&reshdr, sizeof (reshdr), 1);
394*3d8817e4Smiod write_res_id (type);
395*3d8817e4Smiod write_res_id (name);
396*3d8817e4Smiod
397*3d8817e4Smiod res_align_file ();
398*3d8817e4Smiod
399*3d8817e4Smiod write_res_info (resinfo);
400*3d8817e4Smiod res_align_file ();
401*3d8817e4Smiod }
402*3d8817e4Smiod
403*3d8817e4Smiod
404*3d8817e4Smiod /* Write data to file, abort on failure */
405*3d8817e4Smiod static void
write_res_data(data,size,count)406*3d8817e4Smiod write_res_data (data, size, count)
407*3d8817e4Smiod const void *data;
408*3d8817e4Smiod size_t size;
409*3d8817e4Smiod int count;
410*3d8817e4Smiod {
411*3d8817e4Smiod if ((size_t) fwrite (data, size, count, fres) != (size_t) count)
412*3d8817e4Smiod fatal ("%s: could not write to file", filename);
413*3d8817e4Smiod }
414*3d8817e4Smiod
415*3d8817e4Smiod /* Read data from file, abort on failure */
416*3d8817e4Smiod static void
read_res_data(data,size,count)417*3d8817e4Smiod read_res_data (data, size, count)
418*3d8817e4Smiod void *data;
419*3d8817e4Smiod size_t size;
420*3d8817e4Smiod int count;
421*3d8817e4Smiod {
422*3d8817e4Smiod if (fread (data, size, count, fres) != (size_t) count)
423*3d8817e4Smiod fatal ("%s: unexpected end of file", filename);
424*3d8817e4Smiod }
425*3d8817e4Smiod
426*3d8817e4Smiod /* Write a resource id */
427*3d8817e4Smiod static void
write_res_id(id)428*3d8817e4Smiod write_res_id (id)
429*3d8817e4Smiod const struct res_id *id;
430*3d8817e4Smiod {
431*3d8817e4Smiod if (id->named)
432*3d8817e4Smiod {
433*3d8817e4Smiod unsigned long len = id->u.n.length;
434*3d8817e4Smiod unichar null_term = 0;
435*3d8817e4Smiod write_res_data (id->u.n.name, len * sizeof (unichar), 1);
436*3d8817e4Smiod write_res_data (&null_term, sizeof (null_term), 1);
437*3d8817e4Smiod }
438*3d8817e4Smiod else
439*3d8817e4Smiod {
440*3d8817e4Smiod unsigned short i = 0xFFFF;
441*3d8817e4Smiod write_res_data (&i, sizeof (i), 1);
442*3d8817e4Smiod i = id->u.id;
443*3d8817e4Smiod write_res_data (&i, sizeof (i), 1);
444*3d8817e4Smiod }
445*3d8817e4Smiod }
446*3d8817e4Smiod
447*3d8817e4Smiod /* Write resource info */
448*3d8817e4Smiod static void
write_res_info(info)449*3d8817e4Smiod write_res_info (info)
450*3d8817e4Smiod const struct res_res_info *info;
451*3d8817e4Smiod {
452*3d8817e4Smiod write_res_data (&info->version, sizeof (info->version), 1);
453*3d8817e4Smiod write_res_data (&info->memflags, sizeof (info->memflags), 1);
454*3d8817e4Smiod write_res_data (&info->language, sizeof (info->language), 1);
455*3d8817e4Smiod write_res_data (&info->version, sizeof (info->version), 1);
456*3d8817e4Smiod write_res_data (&info->characteristics, sizeof (info->characteristics), 1);
457*3d8817e4Smiod }
458*3d8817e4Smiod
459*3d8817e4Smiod /* read a resource identifier */
460*3d8817e4Smiod void
read_res_id(id)461*3d8817e4Smiod read_res_id (id)
462*3d8817e4Smiod struct res_id *id;
463*3d8817e4Smiod {
464*3d8817e4Smiod unsigned short ord;
465*3d8817e4Smiod unichar *id_s = NULL;
466*3d8817e4Smiod int len;
467*3d8817e4Smiod
468*3d8817e4Smiod read_res_data (&ord, sizeof (ord), 1);
469*3d8817e4Smiod if (ord == 0xFFFF) /* an ordinal id */
470*3d8817e4Smiod {
471*3d8817e4Smiod read_res_data (&ord, sizeof (ord), 1);
472*3d8817e4Smiod id->named = 0;
473*3d8817e4Smiod id->u.id = ord;
474*3d8817e4Smiod }
475*3d8817e4Smiod else
476*3d8817e4Smiod /* named id */
477*3d8817e4Smiod {
478*3d8817e4Smiod if (fseek (fres, -sizeof (ord), SEEK_CUR) != 0)
479*3d8817e4Smiod fatal ("%s: %s: could not seek in file", program_name, filename);
480*3d8817e4Smiod id_s = read_unistring (&len);
481*3d8817e4Smiod id->named = 1;
482*3d8817e4Smiod id->u.n.length = len;
483*3d8817e4Smiod id->u.n.name = id_s;
484*3d8817e4Smiod }
485*3d8817e4Smiod }
486*3d8817e4Smiod
487*3d8817e4Smiod /* Read a null terminated UNICODE string */
488*3d8817e4Smiod static unichar *
read_unistring(len)489*3d8817e4Smiod read_unistring (len)
490*3d8817e4Smiod int *len;
491*3d8817e4Smiod {
492*3d8817e4Smiod unichar *s;
493*3d8817e4Smiod unichar c;
494*3d8817e4Smiod unichar *p;
495*3d8817e4Smiod int l;
496*3d8817e4Smiod
497*3d8817e4Smiod *len = 0;
498*3d8817e4Smiod l = 0;
499*3d8817e4Smiod
500*3d8817e4Smiod /* there are hardly any names longer than 256 characters */
501*3d8817e4Smiod p = s = (unichar *) xmalloc (sizeof (unichar) * 256);
502*3d8817e4Smiod do
503*3d8817e4Smiod {
504*3d8817e4Smiod read_res_data (&c, sizeof (c), 1);
505*3d8817e4Smiod *p++ = c;
506*3d8817e4Smiod if (c != 0)
507*3d8817e4Smiod l++;
508*3d8817e4Smiod }
509*3d8817e4Smiod while (c != 0);
510*3d8817e4Smiod *len = l;
511*3d8817e4Smiod return s;
512*3d8817e4Smiod }
513*3d8817e4Smiod
514*3d8817e4Smiod /* align file on DWORD boundary */
515*3d8817e4Smiod static void
res_align_file(void)516*3d8817e4Smiod res_align_file (void)
517*3d8817e4Smiod {
518*3d8817e4Smiod int pos = ftell (fres);
519*3d8817e4Smiod int skip = ((pos + 3) & ~3) - pos;
520*3d8817e4Smiod if (fseek (fres, skip, SEEK_CUR) != 0)
521*3d8817e4Smiod fatal ("%s: %s: unable to align file", program_name, filename);
522*3d8817e4Smiod }
523*3d8817e4Smiod
524*3d8817e4Smiod /* Check if file is a win32 binary resource file, if so
525*3d8817e4Smiod skip past the null resource. Returns 0 if successful, -1 on
526*3d8817e4Smiod error.
527*3d8817e4Smiod */
528*3d8817e4Smiod static void
skip_null_resource(void)529*3d8817e4Smiod skip_null_resource (void)
530*3d8817e4Smiod {
531*3d8817e4Smiod struct res_hdr reshdr =
532*3d8817e4Smiod {0, 0};
533*3d8817e4Smiod read_res_data (&reshdr, sizeof (reshdr), 1);
534*3d8817e4Smiod if ((reshdr.data_size != 0) || (reshdr.header_size != 0x20))
535*3d8817e4Smiod goto skip_err;
536*3d8817e4Smiod
537*3d8817e4Smiod /* Subtract size of HeaderSize and DataSize */
538*3d8817e4Smiod if (fseek (fres, reshdr.header_size - 8, SEEK_CUR) != 0)
539*3d8817e4Smiod goto skip_err;
540*3d8817e4Smiod
541*3d8817e4Smiod return;
542*3d8817e4Smiod
543*3d8817e4Smiod skip_err:
544*3d8817e4Smiod fprintf (stderr, "%s: %s: Not a valid WIN32 resource file\n", program_name,
545*3d8817e4Smiod filename);
546*3d8817e4Smiod xexit (1);
547*3d8817e4Smiod }
548*3d8817e4Smiod
549*3d8817e4Smiod /* Add a resource to resource directory */
550*3d8817e4Smiod void
res_add_resource(r,type,id,language,dupok)551*3d8817e4Smiod res_add_resource (r, type, id, language, dupok)
552*3d8817e4Smiod struct res_resource *r;
553*3d8817e4Smiod const struct res_id *type;
554*3d8817e4Smiod const struct res_id *id;
555*3d8817e4Smiod int language;
556*3d8817e4Smiod int dupok;
557*3d8817e4Smiod {
558*3d8817e4Smiod struct res_id a[3];
559*3d8817e4Smiod
560*3d8817e4Smiod a[0] = *type;
561*3d8817e4Smiod a[1] = *id;
562*3d8817e4Smiod a[2].named = 0;
563*3d8817e4Smiod a[2].u.id = language;
564*3d8817e4Smiod res_append_resource (&resources, r, 3, a, dupok);
565*3d8817e4Smiod }
566*3d8817e4Smiod
567*3d8817e4Smiod /* Append a resource to resource directory.
568*3d8817e4Smiod This is just copied from define_resource
569*3d8817e4Smiod and modified to add an existing resource.
570*3d8817e4Smiod */
571*3d8817e4Smiod void
res_append_resource(resources,resource,cids,ids,dupok)572*3d8817e4Smiod res_append_resource (resources, resource, cids, ids, dupok)
573*3d8817e4Smiod struct res_directory **resources;
574*3d8817e4Smiod struct res_resource *resource;
575*3d8817e4Smiod int cids;
576*3d8817e4Smiod const struct res_id *ids;
577*3d8817e4Smiod int dupok;
578*3d8817e4Smiod {
579*3d8817e4Smiod struct res_entry *re = NULL;
580*3d8817e4Smiod int i;
581*3d8817e4Smiod
582*3d8817e4Smiod assert (cids > 0);
583*3d8817e4Smiod for (i = 0; i < cids; i++)
584*3d8817e4Smiod {
585*3d8817e4Smiod struct res_entry **pp;
586*3d8817e4Smiod
587*3d8817e4Smiod if (*resources == NULL)
588*3d8817e4Smiod {
589*3d8817e4Smiod static unsigned long timeval;
590*3d8817e4Smiod
591*3d8817e4Smiod /* Use the same timestamp for every resource created in a
592*3d8817e4Smiod single run. */
593*3d8817e4Smiod if (timeval == 0)
594*3d8817e4Smiod timeval = time (NULL);
595*3d8817e4Smiod
596*3d8817e4Smiod *resources = ((struct res_directory *)
597*3d8817e4Smiod res_alloc (sizeof **resources));
598*3d8817e4Smiod (*resources)->characteristics = 0;
599*3d8817e4Smiod (*resources)->time = timeval;
600*3d8817e4Smiod (*resources)->major = 0;
601*3d8817e4Smiod (*resources)->minor = 0;
602*3d8817e4Smiod (*resources)->entries = NULL;
603*3d8817e4Smiod }
604*3d8817e4Smiod
605*3d8817e4Smiod for (pp = &(*resources)->entries; *pp != NULL; pp = &(*pp)->next)
606*3d8817e4Smiod if (res_id_cmp ((*pp)->id, ids[i]) == 0)
607*3d8817e4Smiod break;
608*3d8817e4Smiod
609*3d8817e4Smiod if (*pp != NULL)
610*3d8817e4Smiod re = *pp;
611*3d8817e4Smiod else
612*3d8817e4Smiod {
613*3d8817e4Smiod re = (struct res_entry *) res_alloc (sizeof *re);
614*3d8817e4Smiod re->next = NULL;
615*3d8817e4Smiod re->id = ids[i];
616*3d8817e4Smiod if ((i + 1) < cids)
617*3d8817e4Smiod {
618*3d8817e4Smiod re->subdir = 1;
619*3d8817e4Smiod re->u.dir = NULL;
620*3d8817e4Smiod }
621*3d8817e4Smiod else
622*3d8817e4Smiod {
623*3d8817e4Smiod re->subdir = 0;
624*3d8817e4Smiod re->u.res = NULL;
625*3d8817e4Smiod }
626*3d8817e4Smiod
627*3d8817e4Smiod *pp = re;
628*3d8817e4Smiod }
629*3d8817e4Smiod
630*3d8817e4Smiod if ((i + 1) < cids)
631*3d8817e4Smiod {
632*3d8817e4Smiod if (!re->subdir)
633*3d8817e4Smiod {
634*3d8817e4Smiod fprintf (stderr, "%s: ", program_name);
635*3d8817e4Smiod res_ids_print (stderr, i, ids);
636*3d8817e4Smiod fprintf (stderr, ": expected to be a directory\n");
637*3d8817e4Smiod xexit (1);
638*3d8817e4Smiod }
639*3d8817e4Smiod
640*3d8817e4Smiod resources = &re->u.dir;
641*3d8817e4Smiod }
642*3d8817e4Smiod }
643*3d8817e4Smiod
644*3d8817e4Smiod if (re->subdir)
645*3d8817e4Smiod {
646*3d8817e4Smiod fprintf (stderr, "%s: ", program_name);
647*3d8817e4Smiod res_ids_print (stderr, cids, ids);
648*3d8817e4Smiod fprintf (stderr, ": expected to be a leaf\n");
649*3d8817e4Smiod xexit (1);
650*3d8817e4Smiod }
651*3d8817e4Smiod
652*3d8817e4Smiod if (re->u.res != NULL)
653*3d8817e4Smiod {
654*3d8817e4Smiod if (dupok)
655*3d8817e4Smiod return;
656*3d8817e4Smiod
657*3d8817e4Smiod fprintf (stderr, "%s: warning: ", program_name);
658*3d8817e4Smiod res_ids_print (stderr, cids, ids);
659*3d8817e4Smiod fprintf (stderr, ": duplicate value\n");
660*3d8817e4Smiod }
661*3d8817e4Smiod
662*3d8817e4Smiod re->u.res = resource;
663*3d8817e4Smiod }
664