aboutsummaryrefslogtreecommitdiff
path: root/binutils/resrc.c
diff options
context:
space:
mode:
Diffstat (limited to 'binutils/resrc.c')
-rw-r--r--binutils/resrc.c1624
1 files changed, 1148 insertions, 476 deletions
diff --git a/binutils/resrc.c b/binutils/resrc.c
index 0ec2875..24b423d 100644
--- a/binutils/resrc.c
+++ b/binutils/resrc.c
@@ -2,6 +2,7 @@
Copyright 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2005, 2007
Free Software Foundation, Inc.
Written by Ian Lance Taylor, Cygnus Support.
+ Rewritten by Kai Tietz, Onevision.
This file is part of GNU Binutils.
@@ -25,13 +26,17 @@
#include "sysdep.h"
#include "bfd.h"
+#include "bucomm.h"
#include "libiberty.h"
#include "safe-ctype.h"
-#include "bucomm.h"
#include "windres.h"
#include <assert.h>
+#include <errno.h>
#include <sys/stat.h>
+#ifdef HAVE_UNISTD_H
+#include <unistd.h>
+#endif
#ifdef HAVE_SYS_WAIT_H
#include <sys/wait.h>
@@ -84,11 +89,11 @@
struct icondir
{
/* Width of image. */
- unsigned char width;
+ bfd_byte width;
/* Height of image. */
- unsigned char height;
+ bfd_byte height;
/* Number of colors in image. */
- unsigned char colorcount;
+ bfd_byte colorcount;
union
{
struct
@@ -122,7 +127,7 @@ int rc_lineno;
/* The pipe we are reading from, so that we can close it if we exit. */
-static FILE *cpp_pipe;
+FILE *cpp_pipe;
/* The temporary file used if we're not using popen, so we can delete it
if we exit. */
@@ -135,7 +140,7 @@ static enum {ISTREAM_PIPE, ISTREAM_FILE} istream_type;
/* As we read the rc file, we attach information to this structure. */
-static struct res_directory *resources;
+static rc_res_directory *resources;
/* The number of cursor resources we have written out. */
@@ -147,17 +152,39 @@ static int fonts;
/* Font directory information. */
-struct fontdir *fontdirs;
+rc_fontdir *fontdirs;
/* Resource info to use for fontdirs. */
-struct res_res_info fontdirs_resinfo;
+rc_res_res_info fontdirs_resinfo;
/* The number of icon resources we have written out. */
static int icons;
-/* Local functions. */
+/* The windres target bfd . */
+
+static windres_bfd wrtarget =
+{
+ (bfd *) NULL, (asection *) NULL, WR_KIND_TARGET
+};
+
+/* Local functions for rcdata based resource definitions. */
+
+static void define_font_rcdata (rc_res_id, const rc_res_res_info *,
+ rc_rcdata_item *);
+static void define_icon_rcdata (rc_res_id, const rc_res_res_info *,
+ rc_rcdata_item *);
+static void define_bitmap_rcdata (rc_res_id, const rc_res_res_info *,
+ rc_rcdata_item *);
+static void define_cursor_rcdata (rc_res_id, const rc_res_res_info *,
+ rc_rcdata_item *);
+static void define_fontdir_rcdata (rc_res_id, const rc_res_res_info *,
+ rc_rcdata_item *);
+static void define_messagetable_rcdata (rc_res_id, const rc_res_res_info *,
+ rc_rcdata_item *);
+static rc_uint_type rcdata_copy (const rc_rcdata_item *, bfd_byte *);
+static bfd_byte *rcdata_render_as_buffer (const rc_rcdata_item *, rc_uint_type *);
static int run_cmd (char *, const char *);
static FILE *open_input_stream (char *);
@@ -167,7 +194,7 @@ static void close_input_stream (void);
static void unexpected_eof (const char *);
static int get_word (FILE *, const char *);
static unsigned long get_long (FILE *, const char *);
-static void get_data (FILE *, unsigned char *, unsigned long, const char *);
+static void get_data (FILE *, bfd_byte *, rc_uint_type, const char *);
static void define_fontdirs (void);
/* Run `cmd' and redirect the output to `redir'. */
@@ -375,7 +402,7 @@ look_for_default (char *cmd, const char *prefix, int end_prefix,
/* Read an rc file. */
-struct res_directory *
+rc_res_directory *
read_rc_file (const char *filename, const char *preprocessor,
const char *preprocargs, int language, int use_temp_file)
{
@@ -437,20 +464,20 @@ read_rc_file (const char *filename, const char *preprocessor,
/* First, try looking for a prefixed gcc in the windres
directory, with the same prefix as windres */
- cpp_pipe = look_for_default (cmd, program_name, dash-program_name+1,
+ cpp_pipe = look_for_default (cmd, program_name, dash - program_name + 1,
preprocargs, filename);
}
- if (slash && !cpp_pipe)
+ if (slash && ! cpp_pipe)
{
/* Next, try looking for a gcc in the same directory as
that windres */
- cpp_pipe = look_for_default (cmd, program_name, slash-program_name+1,
+ cpp_pipe = look_for_default (cmd, program_name, slash - program_name + 1,
preprocargs, filename);
}
- if (!cpp_pipe)
+ if (! cpp_pipe)
{
/* Sigh, try the default */
@@ -465,7 +492,6 @@ read_rc_file (const char *filename, const char *preprocessor,
rc_lineno = 1;
if (language != -1)
rcparse_set_language (language);
- yyin = cpp_pipe;
yyparse ();
rcparse_discard_strings ();
@@ -572,24 +598,24 @@ get_long (FILE *e, const char *msg)
/* Read data from a file. This is a wrapper to do error checking. */
static void
-get_data (FILE *e, unsigned char *p, unsigned long c, const char *msg)
+get_data (FILE *e, bfd_byte *p, rc_uint_type c, const char *msg)
{
- unsigned long got;
+ rc_uint_type got; // $$$d
- got = fread (p, 1, c, e);
+ got = (rc_uint_type) fread (p, 1, c, e);
if (got == c)
return;
- fatal (_("%s: read of %lu returned %lu"), msg, c, got);
+ fatal (_("%s: read of %lu returned %lu"), msg, (long) c, (long) got);
}
/* Define an accelerator resource. */
void
-define_accelerator (struct res_id id, const struct res_res_info *resinfo,
- struct accelerator *data)
+define_accelerator (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_accelerator *data)
{
- struct res_resource *r;
+ rc_res_resource *r;
r = define_standard_resource (&resources, RT_ACCELERATOR, id,
resinfo->language, 0);
@@ -605,15 +631,15 @@ define_accelerator (struct res_id id, const struct res_res_info *resinfo,
#define BITMAP_SKIP (14)
void
-define_bitmap (struct res_id id, const struct res_res_info *resinfo,
+define_bitmap (rc_res_id id, const rc_res_res_info *resinfo,
const char *filename)
{
FILE *e;
char *real_filename;
struct stat s;
- unsigned char *data;
- int i;
- struct res_resource *r;
+ bfd_byte *data;
+ rc_uint_type i;
+ rc_res_resource *r;
e = open_file_search (filename, FOPEN_RB, "bitmap file", &real_filename);
@@ -621,7 +647,7 @@ define_bitmap (struct res_id id, const struct res_res_info *resinfo,
fatal (_("stat failed on bitmap file `%s': %s"), real_filename,
strerror (errno));
- data = (unsigned char *) res_alloc (s.st_size - BITMAP_SKIP);
+ data = (bfd_byte *) res_alloc (s.st_size - BITMAP_SKIP);
for (i = 0; i < BITMAP_SKIP; i++)
getc (e);
@@ -647,7 +673,7 @@ define_bitmap (struct res_id id, const struct res_res_info *resinfo,
select one of the actual cursors. */
void
-define_cursor (struct res_id id, const struct res_res_info *resinfo,
+define_cursor (rc_res_id id, const rc_res_res_info *resinfo,
const char *filename)
{
FILE *e;
@@ -655,8 +681,8 @@ define_cursor (struct res_id id, const struct res_res_info *resinfo,
int type, count, i;
struct icondir *icondirs;
int first_cursor;
- struct res_resource *r;
- struct group_cursor *first, **pp;
+ rc_res_resource *r;
+ rc_group_cursor *first, **pp;
e = open_file_search (filename, FOPEN_RB, "cursor file", &real_filename);
@@ -696,19 +722,19 @@ define_cursor (struct res_id id, const struct res_res_info *resinfo,
for (i = 0; i < count; i++)
{
- unsigned char *data;
- struct res_id name;
- struct cursor *c;
+ bfd_byte *data;
+ rc_res_id name;
+ rc_cursor *c;
if (fseek (e, icondirs[i].offset, SEEK_SET) != 0)
fatal (_("%s: fseek to %lu failed: %s"), real_filename,
icondirs[i].offset, strerror (errno));
- data = (unsigned char *) res_alloc (icondirs[i].bytes);
+ data = (bfd_byte *) res_alloc (icondirs[i].bytes);
get_data (e, data, icondirs[i].bytes, real_filename);
- c = (struct cursor *) res_alloc (sizeof *c);
+ c = (rc_cursor *) res_alloc (sizeof (rc_cursor));
c->xhotspot = icondirs[i].u.cursor.xhotspot;
c->yhotspot = icondirs[i].u.cursor.yhotspot;
c->length = icondirs[i].bytes;
@@ -735,9 +761,9 @@ define_cursor (struct res_id id, const struct res_res_info *resinfo,
pp = &first;
for (i = 0; i < count; i++)
{
- struct group_cursor *cg;
+ rc_group_cursor *cg;
- cg = (struct group_cursor *) res_alloc (sizeof *cg);
+ cg = (rc_group_cursor *) res_alloc (sizeof (rc_group_cursor));
cg->next = NULL;
cg->width = icondirs[i].width;
cg->height = 2 * icondirs[i].height;
@@ -765,13 +791,13 @@ define_cursor (struct res_id id, const struct res_res_info *resinfo,
/* Define a dialog resource. */
void
-define_dialog (struct res_id id, const struct res_res_info *resinfo,
- const struct dialog *dialog)
+define_dialog (rc_res_id id, const rc_res_res_info *resinfo,
+ const rc_dialog *dialog)
{
- struct dialog *copy;
- struct res_resource *r;
+ rc_dialog *copy;
+ rc_res_resource *r;
- copy = (struct dialog *) res_alloc (sizeof *copy);
+ copy = (rc_dialog *) res_alloc (sizeof *copy);
*copy = *dialog;
r = define_standard_resource (&resources, RT_DIALOG, id,
@@ -784,15 +810,15 @@ define_dialog (struct res_id id, const struct res_res_info *resinfo,
/* Define a dialog control. This does not define a resource, but
merely allocates and fills in a structure. */
-struct dialog_control *
-define_control (const struct res_id iid, unsigned long id, unsigned long x,
- unsigned long y, unsigned long width, unsigned long height,
- unsigned long class, unsigned long style,
- unsigned long exstyle)
+rc_dialog_control *
+define_control (const rc_res_id iid, rc_uint_type id, rc_uint_type x,
+ rc_uint_type y, rc_uint_type width, rc_uint_type height,
+ const rc_res_id class, rc_uint_type style,
+ rc_uint_type exstyle)
{
- struct dialog_control *n;
+ rc_dialog_control *n;
- n = (struct dialog_control *) res_alloc (sizeof *n);
+ n = (rc_dialog_control *) res_alloc (sizeof (rc_dialog_control));
n->next = NULL;
n->id = id;
n->style = style;
@@ -801,8 +827,7 @@ define_control (const struct res_id iid, unsigned long id, unsigned long x,
n->y = y;
n->width = width;
n->height = height;
- n->class.named = 0;
- n->class.u.id = class;
+ n->class = class;
n->text = iid;
n->data = NULL;
n->help = 0;
@@ -810,23 +835,26 @@ define_control (const struct res_id iid, unsigned long id, unsigned long x,
return n;
}
-struct dialog_control *
-define_icon_control (struct res_id iid, unsigned long id, unsigned long x,
- unsigned long y, unsigned long style,
- unsigned long exstyle, unsigned long help,
- struct rcdata_item *data, struct dialog_ex *ex)
+rc_dialog_control *
+define_icon_control (rc_res_id iid, rc_uint_type id, rc_uint_type x,
+ rc_uint_type y, rc_uint_type style,
+ rc_uint_type exstyle, rc_uint_type help,
+ rc_rcdata_item *data, rc_dialog_ex *ex)
{
- struct dialog_control *n;
- struct res_id tid;
+ rc_dialog_control *n;
+ rc_res_id tid;
+ rc_res_id cid;
if (style == 0)
style = SS_ICON | WS_CHILD | WS_VISIBLE;
res_string_to_id (&tid, "");
- n = define_control (tid, id, x, y, 0, 0, CTL_STATIC, style, exstyle);
+ cid.named = 0;
+ cid.u.id = CTL_STATIC;
+ n = define_control (tid, id, x, y, 0, 0, cid, style, exstyle);
n->text = iid;
- if (help && !ex)
+ if (help && ! ex)
rcparse_warning (_("help ID requires DIALOGEX"));
- if (data && !ex)
+ if (data && ! ex)
rcparse_warning (_("control data requires DIALOGEX"));
n->help = help;
n->data = data;
@@ -837,20 +865,20 @@ define_icon_control (struct res_id iid, unsigned long id, unsigned long x,
/* Define a font resource. */
void
-define_font (struct res_id id, const struct res_res_info *resinfo,
+define_font (rc_res_id id, const rc_res_res_info *resinfo,
const char *filename)
{
FILE *e;
char *real_filename;
struct stat s;
- unsigned char *data;
- struct res_resource *r;
+ bfd_byte *data;
+ rc_res_resource *r;
long offset;
long fontdatalength;
- unsigned char *fontdata;
- struct fontdir *fd;
+ bfd_byte *fontdata;
+ rc_fontdir *fd;
const char *device, *face;
- struct fontdir **pp;
+ rc_fontdir **pp;
e = open_file_search (filename, FOPEN_RB, "font file", &real_filename);
@@ -858,7 +886,7 @@ define_font (struct res_id id, const struct res_res_info *resinfo,
fatal (_("stat failed on font file `%s': %s"), real_filename,
strerror (errno));
- data = (unsigned char *) res_alloc (s.st_size);
+ data = (bfd_byte *) res_alloc (s.st_size);
get_data (e, data, s.st_size, real_filename);
@@ -899,12 +927,12 @@ define_font (struct res_id id, const struct res_res_info *resinfo,
++fonts;
fontdatalength = 58 + strlen (device) + strlen (face);
- fontdata = (unsigned char *) res_alloc (fontdatalength);
+ fontdata = (bfd_byte *) res_alloc (fontdatalength);
memcpy (fontdata, data, 56);
strcpy ((char *) fontdata + 56, device);
strcpy ((char *) fontdata + 57 + strlen (device), face);
- fd = (struct fontdir *) res_alloc (sizeof *fd);
+ fd = (rc_fontdir *) res_alloc (sizeof (rc_fontdir));
fd->next = NULL;
fd->index = fonts;
fd->length = fontdatalength;
@@ -919,14 +947,33 @@ define_font (struct res_id id, const struct res_res_info *resinfo,
fontdirs_resinfo = *resinfo;
}
+static void
+define_font_rcdata (rc_res_id id,const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ r = define_standard_resource (&resources, RT_FONT, id,
+ resinfo->language, 0);
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ r->type = RES_TYPE_FONT;
+ r->u.data.length = len_data;
+ r->u.data.data = pb_data;
+ r->res_info = *resinfo;
+}
+
/* Define the fontdirs resource. This is called after the entire rc
file has been parsed, if any font resources were seen. */
static void
define_fontdirs (void)
{
- struct res_resource *r;
- struct res_id id;
+ rc_res_resource *r;
+ rc_res_id id;
id.named = 0;
id.u.id = 1;
@@ -938,6 +985,89 @@ define_fontdirs (void)
r->res_info = fontdirs_resinfo;
}
+static bfd_byte *
+rcdata_render_as_buffer (const rc_rcdata_item *data, rc_uint_type *plen)
+{
+ const rc_rcdata_item *d;
+ bfd_byte *ret = NULL, *pret;
+ rc_uint_type len = 0;
+
+ for (d = data; d != NULL; d = d->next)
+ len += rcdata_copy (d, NULL);
+ if (len != 0)
+ {
+ ret = pret = (bfd_byte *) res_alloc (len);
+ for (d = data; d != NULL; d = d->next)
+ pret += rcdata_copy (d, pret);
+ }
+ if (plen)
+ *plen = len;
+ return ret;
+}
+
+static void
+define_fontdir_rcdata (rc_res_id id,const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_fontdir *fd, *fd_first, *fd_cur;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+ rc_uint_type c;
+
+ fd_cur = fd_first = NULL;
+ r = define_standard_resource (&resources, RT_FONTDIR, id, 0x409, 0);
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ if (pb_data)
+ {
+ rc_uint_type off = 2;
+ c = windres_get_16 (&wrtarget, pb_data, len_data);
+ for (; c > 0; c--)
+ {
+ size_t len;
+ rc_uint_type safe_pos = off;
+ const struct bin_fontdir_item *bfi;
+
+ bfi = (const struct bin_fontdir_item *) pb_data + off;
+ fd = (rc_fontdir *) res_alloc (sizeof (rc_fontdir));
+ fd->index = windres_get_16 (&wrtarget, bfi->index, len_data - off);
+ fd->data = pb_data + off;
+ off += 56;
+ len = strlen ((char *) bfi->device_name) + 1;
+ off += (rc_uint_type) len;
+ off += (rc_uint_type) strlen ((char *) bfi->device_name + len) + 1;
+ fd->length = (off - safe_pos);
+ fd->next = NULL;
+ if (fd_first == NULL)
+ fd_first = fd;
+ else
+ fd_cur->next = fd;
+ fd_cur = fd;
+ }
+ }
+ r->type = RES_TYPE_FONTDIR;
+ r->u.fontdir = fd_first;
+ r->res_info = *resinfo;
+}
+
+static void define_messagetable_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ r = define_standard_resource (&resources, RT_MESSAGETABLE, id, resinfo->language, 0);
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+ r->type = RES_TYPE_MESSAGETABLE;
+ r->u.data.length = len_data;
+ r->u.data.data = pb_data;
+ r->res_info = *resinfo;
+}
+
/* Define an icon resource. An icon file may contain a set of
bitmaps, each representing the same icon at various different
resolutions. They each get written out with a different ID. The
@@ -945,7 +1075,7 @@ define_fontdirs (void)
select one of the actual icon bitmaps. */
void
-define_icon (struct res_id id, const struct res_res_info *resinfo,
+define_icon (rc_res_id id, const rc_res_res_info *resinfo,
const char *filename)
{
FILE *e;
@@ -953,8 +1083,8 @@ define_icon (struct res_id id, const struct res_res_info *resinfo,
int type, count, i;
struct icondir *icondirs;
int first_icon;
- struct res_resource *r;
- struct group_icon *first, **pp;
+ rc_res_resource *r;
+ rc_group_icon *first, **pp;
e = open_file_search (filename, FOPEN_RB, "icon file", &real_filename);
@@ -993,14 +1123,14 @@ define_icon (struct res_id id, const struct res_res_info *resinfo,
for (i = 0; i < count; i++)
{
- unsigned char *data;
- struct res_id name;
+ bfd_byte *data;
+ rc_res_id name;
if (fseek (e, icondirs[i].offset, SEEK_SET) != 0)
fatal (_("%s: fseek to %lu failed: %s"), real_filename,
icondirs[i].offset, strerror (errno));
- data = (unsigned char *) res_alloc (icondirs[i].bytes);
+ data = (bfd_byte *) res_alloc (icondirs[i].bytes);
get_data (e, data, icondirs[i].bytes, real_filename);
@@ -1026,13 +1156,13 @@ define_icon (struct res_id id, const struct res_res_info *resinfo,
pp = &first;
for (i = 0; i < count; i++)
{
- struct group_icon *cg;
+ rc_group_icon *cg;
/* For some reason, at least in some files the planes and bits
are zero. We instead set them from the color. This is
copied from rcl. */
- cg = (struct group_icon *) res_alloc (sizeof *cg);
+ cg = (rc_group_icon *) res_alloc (sizeof (rc_group_icon));
cg->next = NULL;
cg->width = icondirs[i].width;
cg->height = icondirs[i].height;
@@ -1069,16 +1199,180 @@ define_icon (struct res_id id, const struct res_res_info *resinfo,
r->res_info = *resinfo;
}
+static void
+define_group_icon_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_group_icon *cg, *first, *cur;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ cur = NULL;
+ first = NULL;
+
+ while (len_data >= 6)
+ {
+ int c, i;
+ unsigned short type;
+ type = windres_get_16 (&wrtarget, pb_data + 2, len_data - 2);
+ if (type != 1)
+ fatal (_("unexpected group icon type %d"), type);
+ c = windres_get_16 (&wrtarget, pb_data + 4, len_data - 4);
+ len_data -= 6;
+ pb_data += 6;
+
+ for (i = 0; i < c; i++)
+ {
+ if (len_data < 14)
+ fatal ("too small group icon rcdata");
+ cg = (rc_group_icon *) res_alloc (sizeof (rc_group_icon));
+ cg->next = NULL;
+ cg->width = pb_data[0];
+ cg->height = pb_data[1];
+ cg->colors = pb_data[2];
+ cg->planes = windres_get_16 (&wrtarget, pb_data + 4, len_data - 4);
+ cg->bits = windres_get_16 (&wrtarget, pb_data + 6, len_data - 6);
+ cg->bytes = windres_get_32 (&wrtarget, pb_data + 8, len_data - 8);
+ cg->index = windres_get_16 (&wrtarget, pb_data + 12, len_data - 12);
+ if (! first)
+ first = cg;
+ else
+ cur->next = cg;
+ cur = cg;
+ pb_data += 14;
+ len_data -= 14;
+ }
+ }
+ r = define_standard_resource (&resources, RT_GROUP_ICON, id,
+ resinfo->language, 0);
+ r->type = RES_TYPE_GROUP_ICON;
+ r->u.group_icon = first;
+ r->res_info = *resinfo;
+}
+
+static void
+define_group_cursor_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_group_cursor *cg, *first, *cur;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ first = cur = NULL;
+
+ while (len_data >= 6)
+ {
+ int c, i;
+ unsigned short type;
+ type = windres_get_16 (&wrtarget, pb_data + 2, len_data - 2);
+ if (type != 2)
+ fatal (_("unexpected group cursor type %d"), type);
+ c = windres_get_16 (&wrtarget, pb_data + 4, len_data - 4);
+ len_data -= 6;
+ pb_data += 6;
+
+ for (i = 0; i < c; i++)
+ {
+ if (len_data < 14)
+ fatal ("too small group icon rcdata");
+ cg = (rc_group_cursor *) res_alloc (sizeof (rc_group_cursor));
+ cg->next = NULL;
+ cg->width = windres_get_16 (&wrtarget, pb_data, len_data);
+ cg->height = windres_get_16 (&wrtarget, pb_data + 2, len_data - 2);
+ cg->planes = windres_get_16 (&wrtarget, pb_data + 4, len_data - 4);
+ cg->bits = windres_get_16 (&wrtarget, pb_data + 6, len_data - 6);
+ cg->bytes = windres_get_32 (&wrtarget, pb_data + 8, len_data - 8);
+ cg->index = windres_get_16 (&wrtarget, pb_data + 12, len_data - 12);
+ if (! first)
+ first = cg;
+ else
+ cur->next = cg;
+ cur = cg;
+ pb_data += 14;
+ len_data -= 14;
+ }
+ }
+
+ r = define_standard_resource (&resources, RT_GROUP_ICON, id,
+ resinfo->language, 0);
+ r->type = RES_TYPE_GROUP_CURSOR;
+ r->u.group_cursor = first;
+ r->res_info = *resinfo;
+}
+
+static void
+define_cursor_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_cursor *c;
+ rc_res_resource *r;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ c = (rc_cursor *) res_alloc (sizeof (rc_cursor));
+ c->xhotspot = windres_get_16 (&wrtarget, pb_data, len_data);
+ c->yhotspot = windres_get_16 (&wrtarget, pb_data + 2, len_data - 2);
+ c->length = len_data - BIN_CURSOR_SIZE;
+ c->data = (const bfd_byte *) (data + BIN_CURSOR_SIZE);
+
+ r = define_standard_resource (&resources, RT_CURSOR, id, resinfo->language, 0);
+ r->type = RES_TYPE_CURSOR;
+ r->u.cursor = c;
+ r->res_info = *resinfo;
+}
+
+static void
+define_bitmap_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ r = define_standard_resource (&resources, RT_BITMAP, id, resinfo->language, 0);
+ r->type = RES_TYPE_BITMAP;
+ r->u.data.length = len_data;
+ r->u.data.data = pb_data;
+ r->res_info = *resinfo;
+}
+
+static void
+define_icon_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
+{
+ rc_res_resource *r;
+ rc_uint_type len_data;
+ bfd_byte *pb_data;
+
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+
+ r = define_standard_resource (&resources, RT_ICON, id, resinfo->language, 0);
+ r->type = RES_TYPE_ICON;
+ r->u.data.length = len_data;
+ r->u.data.data = pb_data;
+ r->res_info = *resinfo;
+}
+
/* Define a menu resource. */
void
-define_menu (struct res_id id, const struct res_res_info *resinfo,
- struct menuitem *menuitems)
+define_menu (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_menuitem *menuitems)
{
- struct menu *m;
- struct res_resource *r;
+ rc_menu *m;
+ rc_res_resource *r;
- m = (struct menu *) res_alloc (sizeof *m);
+ m = (rc_menu *) res_alloc (sizeof (rc_menu));
m->items = menuitems;
m->help = 0;
@@ -1091,22 +1385,19 @@ define_menu (struct res_id id, const struct res_res_info *resinfo,
/* Define a menu item. This does not define a resource, but merely
allocates and fills in a structure. */
-struct menuitem *
-define_menuitem (const char *text, int menuid, unsigned long type,
- unsigned long state, unsigned long help,
- struct menuitem *menuitems)
+rc_menuitem *
+define_menuitem (const unichar *text, rc_uint_type menuid, rc_uint_type type,
+ rc_uint_type state, rc_uint_type help,
+ rc_menuitem *menuitems)
{
- struct menuitem *mi;
+ rc_menuitem *mi;
- mi = (struct menuitem *) res_alloc (sizeof *mi);
+ mi = (rc_menuitem *) res_alloc (sizeof (rc_menuitem));
mi->next = NULL;
mi->type = type;
mi->state = state;
mi->id = menuid;
- if (text == NULL)
- mi->text = NULL;
- else
- unicode_from_ascii ((int *) NULL, &mi->text, text);
+ mi->text = unichar_dup (text);
mi->help = help;
mi->popup = menuitems;
return mi;
@@ -1115,14 +1406,14 @@ define_menuitem (const char *text, int menuid, unsigned long type,
/* Define a messagetable resource. */
void
-define_messagetable (struct res_id id, const struct res_res_info *resinfo,
+define_messagetable (rc_res_id id, const rc_res_res_info *resinfo,
const char *filename)
{
FILE *e;
char *real_filename;
struct stat s;
- unsigned char *data;
- struct res_resource *r;
+ bfd_byte *data;
+ rc_res_resource *r;
e = open_file_search (filename, FOPEN_RB, "messagetable file",
&real_filename);
@@ -1131,7 +1422,7 @@ define_messagetable (struct res_id id, const struct res_res_info *resinfo,
fatal (_("stat failed on bitmap file `%s': %s"), real_filename,
strerror (errno));
- data = (unsigned char *) res_alloc (s.st_size);
+ data = (bfd_byte *) res_alloc (s.st_size);
get_data (e, data, s.st_size, real_filename);
@@ -1150,10 +1441,10 @@ define_messagetable (struct res_id id, const struct res_res_info *resinfo,
/* Define an rcdata resource. */
void
-define_rcdata (struct res_id id, const struct res_res_info *resinfo,
- struct rcdata_item *data)
+define_rcdata (rc_res_id id, const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
{
- struct res_resource *r;
+ rc_res_resource *r;
r = define_standard_resource (&resources, RT_RCDATA, id,
resinfo->language, 0);
@@ -1164,13 +1455,13 @@ define_rcdata (struct res_id id, const struct res_res_info *resinfo,
/* Create an rcdata item holding a string. */
-struct rcdata_item *
-define_rcdata_string (const char *string, unsigned long len)
+rc_rcdata_item *
+define_rcdata_string (const char *string, rc_uint_type len)
{
- struct rcdata_item *ri;
+ rc_rcdata_item *ri;
char *s;
- ri = (struct rcdata_item *) res_alloc (sizeof *ri);
+ ri = (rc_rcdata_item *) res_alloc (sizeof (rc_rcdata_item));
ri->next = NULL;
ri->type = RCDATA_STRING;
ri->u.string.length = len;
@@ -1181,14 +1472,33 @@ define_rcdata_string (const char *string, unsigned long len)
return ri;
}
+/* Create an rcdata item holding a unicode string. */
+
+rc_rcdata_item *
+define_rcdata_unistring (const unichar *string, rc_uint_type len)
+{
+ rc_rcdata_item *ri;
+ unichar *s;
+
+ ri = (rc_rcdata_item *) res_alloc (sizeof (rc_rcdata_item));
+ ri->next = NULL;
+ ri->type = RCDATA_WSTRING;
+ ri->u.wstring.length = len;
+ s = (unichar *) res_alloc (len * sizeof (unichar));
+ memcpy (s, string, len * sizeof (unichar));
+ ri->u.wstring.w = s;
+
+ return ri;
+}
+
/* Create an rcdata item holding a number. */
-struct rcdata_item *
-define_rcdata_number (unsigned long val, int dword)
+rc_rcdata_item *
+define_rcdata_number (rc_uint_type val, int dword)
{
- struct rcdata_item *ri;
+ rc_rcdata_item *ri;
- ri = (struct rcdata_item *) res_alloc (sizeof *ri);
+ ri = (rc_rcdata_item *) res_alloc (sizeof (rc_rcdata_item));
ri->next = NULL;
ri->type = dword ? RCDATA_DWORD : RCDATA_WORD;
ri->u.word = val;
@@ -1200,11 +1510,11 @@ define_rcdata_number (unsigned long val, int dword)
which appears in a STRINGTABLE statement. */
void
-define_stringtable (const struct res_res_info *resinfo,
- unsigned long stringid, const char *string)
+define_stringtable (const rc_res_res_info *resinfo,
+ rc_uint_type stringid, const unichar *string)
{
- struct res_id id;
- struct res_resource *r;
+ rc_res_id id;
+ rc_res_resource *r;
id.named = 0;
id.u.id = (stringid >> 4) + 1;
@@ -1216,8 +1526,8 @@ define_stringtable (const struct res_res_info *resinfo,
int i;
r->type = RES_TYPE_STRINGTABLE;
- r->u.stringtable = ((struct stringtable *)
- res_alloc (sizeof (struct stringtable)));
+ r->u.stringtable = ((rc_stringtable *)
+ res_alloc (sizeof (rc_stringtable)));
for (i = 0; i < 16; i++)
{
r->u.stringtable->strings[i].length = 0;
@@ -1227,21 +1537,79 @@ define_stringtable (const struct res_res_info *resinfo,
r->res_info = *resinfo;
}
- unicode_from_ascii (&r->u.stringtable->strings[stringid & 0xf].length,
- &r->u.stringtable->strings[stringid & 0xf].string,
- string);
+ r->u.stringtable->strings[stringid & 0xf].length = unichar_len (string);
+ r->u.stringtable->strings[stringid & 0xf].string = unichar_dup (string);
+}
+
+void
+define_toolbar (rc_res_id id, rc_res_res_info *resinfo, rc_uint_type width, rc_uint_type height,
+ rc_toolbar_item *items)
+{
+ rc_toolbar *t;
+ rc_res_resource *r;
+
+ t = (rc_toolbar *) res_alloc (sizeof (rc_toolbar));
+ t->button_width = width;
+ t->button_height = height;
+ t->nitems = 0;
+ t->items = items;
+ while (items != NULL)
+ {
+ t->nitems+=1;
+ items = items->next;
+ }
+ r = define_standard_resource (&resources, RT_TOOLBAR, id, resinfo->language, 0);
+ r->type = RES_TYPE_TOOLBAR;
+ r->u.toolbar = t;
+ r->res_info = *resinfo;
}
/* Define a user data resource where the data is in the rc file. */
void
-define_user_data (struct res_id id, struct res_id type,
- const struct res_res_info *resinfo,
- struct rcdata_item *data)
+define_user_data (rc_res_id id, rc_res_id type,
+ const rc_res_res_info *resinfo,
+ rc_rcdata_item *data)
{
- struct res_id ids[3];
- struct res_resource *r;
+ rc_res_id ids[3];
+ rc_res_resource *r;
+ bfd_byte *pb_data;
+ rc_uint_type len_data;
+ /* We have to check if the binary data is parsed specially. */
+ if (type.named == 0)
+ {
+ switch (type.u.id)
+ {
+ case RT_FONTDIR:
+ define_fontdir_rcdata (id, resinfo, data);
+ return;
+ case RT_FONT:
+ define_font_rcdata (id, resinfo, data);
+ return;
+ case RT_ICON:
+ define_icon_rcdata (id, resinfo, data);
+ return;
+ case RT_BITMAP:
+ define_bitmap_rcdata (id, resinfo, data);
+ return;
+ case RT_CURSOR:
+ define_cursor_rcdata (id, resinfo, data);
+ return;
+ case RT_GROUP_ICON:
+ define_group_icon_rcdata (id, resinfo, data);
+ return;
+ case RT_GROUP_CURSOR:
+ define_group_cursor_rcdata (id, resinfo, data);
+ return;
+ case RT_MESSAGETABLE:
+ define_messagetable_rcdata (id, resinfo, data);
+ return;
+ default:
+ /* Treat as normal user-data. */
+ break;
+ }
+ }
ids[0] = type;
ids[1] = id;
ids[2].named = 0;
@@ -1249,19 +1617,25 @@ define_user_data (struct res_id id, struct res_id type,
r = define_resource (& resources, 3, ids, 0);
r->type = RES_TYPE_USERDATA;
- r->u.userdata = data;
+ r->u.userdata = ((rc_rcdata_item *)
+ res_alloc (sizeof (rc_rcdata_item)));
+ r->u.userdata->next = NULL;
+ r->u.userdata->type = RCDATA_BUFFER;
+ pb_data = rcdata_render_as_buffer (data, &len_data);
+ r->u.userdata->u.buffer.length = len_data;
+ r->u.userdata->u.buffer.data = pb_data;
r->res_info = *resinfo;
}
void
-define_rcdata_file (struct res_id id, const struct res_res_info *resinfo,
+define_rcdata_file (rc_res_id id, const rc_res_res_info *resinfo,
const char *filename)
{
- struct rcdata_item *ri;
+ rc_rcdata_item *ri;
FILE *e;
char *real_filename;
struct stat s;
- unsigned char *data;
+ bfd_byte *data;
e = open_file_search (filename, FOPEN_RB, "file", &real_filename);
@@ -1270,14 +1644,14 @@ define_rcdata_file (struct res_id id, const struct res_res_info *resinfo,
fatal (_("stat failed on file `%s': %s"), real_filename,
strerror (errno));
- data = (unsigned char *) res_alloc (s.st_size);
+ data = (bfd_byte *) res_alloc (s.st_size);
get_data (e, data, s.st_size, real_filename);
fclose (e);
free (real_filename);
- ri = (struct rcdata_item *) res_alloc (sizeof *ri);
+ ri = (rc_rcdata_item *) res_alloc (sizeof (rc_rcdata_item));
ri->next = NULL;
ri->type = RCDATA_BUFFER;
ri->u.buffer.length = s.st_size;
@@ -1289,15 +1663,15 @@ define_rcdata_file (struct res_id id, const struct res_res_info *resinfo,
/* Define a user data resource where the data is in a file. */
void
-define_user_file (struct res_id id, struct res_id type,
- const struct res_res_info *resinfo, const char *filename)
+define_user_file (rc_res_id id, rc_res_id type,
+ const rc_res_res_info *resinfo, const char *filename)
{
FILE *e;
char *real_filename;
struct stat s;
- unsigned char *data;
- struct res_id ids[3];
- struct res_resource *r;
+ bfd_byte *data;
+ rc_res_id ids[3];
+ rc_res_resource *r;
e = open_file_search (filename, FOPEN_RB, "file", &real_filename);
@@ -1305,7 +1679,7 @@ define_user_file (struct res_id id, struct res_id type,
fatal (_("stat failed on file `%s': %s"), real_filename,
strerror (errno));
- data = (unsigned char *) res_alloc (s.st_size);
+ data = (bfd_byte *) res_alloc (s.st_size);
get_data (e, data, s.st_size, real_filename);
@@ -1319,8 +1693,8 @@ define_user_file (struct res_id id, struct res_id type,
r = define_resource (&resources, 3, ids, 0);
r->type = RES_TYPE_USERDATA;
- r->u.userdata = ((struct rcdata_item *)
- res_alloc (sizeof (struct rcdata_item)));
+ r->u.userdata = ((rc_rcdata_item *)
+ res_alloc (sizeof (rc_rcdata_item)));
r->u.userdata->next = NULL;
r->u.userdata->type = RCDATA_BUFFER;
r->u.userdata->u.buffer.length = s.st_size;
@@ -1331,16 +1705,16 @@ define_user_file (struct res_id id, struct res_id type,
/* Define a versioninfo resource. */
void
-define_versioninfo (struct res_id id, int language,
- struct fixed_versioninfo *fixedverinfo,
- struct ver_info *verinfo)
+define_versioninfo (rc_res_id id, rc_uint_type language,
+ rc_fixed_versioninfo *fixedverinfo,
+ rc_ver_info *verinfo)
{
- struct res_resource *r;
+ rc_res_resource *r;
r = define_standard_resource (&resources, RT_VERSION, id, language, 0);
r->type = RES_TYPE_VERSIONINFO;
- r->u.versioninfo = ((struct versioninfo *)
- res_alloc (sizeof (struct versioninfo)));
+ r->u.versioninfo = ((rc_versioninfo *)
+ res_alloc (sizeof (rc_versioninfo)));
r->u.versioninfo->fixed = fixedverinfo;
r->u.versioninfo->var = verinfo;
r->res_info.language = language;
@@ -1348,16 +1722,16 @@ define_versioninfo (struct res_id id, int language,
/* Add string version info to a list of version information. */
-struct ver_info *
-append_ver_stringfileinfo (struct ver_info *verinfo, const char *language,
- struct ver_stringinfo *strings)
+rc_ver_info *
+append_ver_stringfileinfo (rc_ver_info *verinfo, const char *language,
+ rc_ver_stringinfo *strings)
{
- struct ver_info *vi, **pp;
+ rc_ver_info *vi, **pp;
- vi = (struct ver_info *) res_alloc (sizeof *vi);
+ vi = (rc_ver_info *) res_alloc (sizeof (rc_ver_info));
vi->next = NULL;
vi->type = VERINFO_STRING;
- unicode_from_ascii ((int *) NULL, &vi->u.string.language, language);
+ unicode_from_ascii ((rc_uint_type *) NULL, &vi->u.string.language, language);
vi->u.string.strings = strings;
for (pp = &verinfo; *pp != NULL; pp = &(*pp)->next)
@@ -1369,16 +1743,16 @@ append_ver_stringfileinfo (struct ver_info *verinfo, const char *language,
/* Add variable version info to a list of version information. */
-struct ver_info *
-append_ver_varfileinfo (struct ver_info *verinfo, const char *key,
- struct ver_varinfo *var)
+rc_ver_info *
+append_ver_varfileinfo (rc_ver_info *verinfo, const unichar *key,
+ rc_ver_varinfo *var)
{
- struct ver_info *vi, **pp;
+ rc_ver_info *vi, **pp;
- vi = (struct ver_info *) res_alloc (sizeof *vi);
+ vi = (rc_ver_info *) res_alloc (sizeof *vi);
vi->next = NULL;
vi->type = VERINFO_VAR;
- unicode_from_ascii ((int *) NULL, &vi->u.var.key, key);
+ vi->u.var.key = unichar_dup (key);
vi->u.var.var = var;
for (pp = &verinfo; *pp != NULL; pp = &(*pp)->next)
@@ -1390,16 +1764,16 @@ append_ver_varfileinfo (struct ver_info *verinfo, const char *key,
/* Append version string information to a list. */
-struct ver_stringinfo *
-append_verval (struct ver_stringinfo *strings, const char *key,
- const char *value)
+rc_ver_stringinfo *
+append_verval (rc_ver_stringinfo *strings, const unichar *key,
+ const unichar *value)
{
- struct ver_stringinfo *vs, **pp;
+ rc_ver_stringinfo *vs, **pp;
- vs = (struct ver_stringinfo *) res_alloc (sizeof *vs);
+ vs = (rc_ver_stringinfo *) res_alloc (sizeof (rc_ver_stringinfo));
vs->next = NULL;
- unicode_from_ascii ((int *) NULL, &vs->key, key);
- unicode_from_ascii ((int *) NULL, &vs->value, value);
+ vs->key = unichar_dup (key);
+ vs->value = unichar_dup (value);
for (pp = &strings; *pp != NULL; pp = &(*pp)->next)
;
@@ -1410,13 +1784,13 @@ append_verval (struct ver_stringinfo *strings, const char *key,
/* Append version variable information to a list. */
-struct ver_varinfo *
-append_vertrans (struct ver_varinfo *var, unsigned long language,
- unsigned long charset)
+rc_ver_varinfo *
+append_vertrans (rc_ver_varinfo *var, rc_uint_type language,
+ rc_uint_type charset)
{
- struct ver_varinfo *vv, **pp;
+ rc_ver_varinfo *vv, **pp;
- vv = (struct ver_varinfo *) res_alloc (sizeof *vv);
+ vv = (rc_ver_varinfo *) res_alloc (sizeof (rc_ver_varinfo));
vv->next = NULL;
vv->language = language;
vv->charset = charset;
@@ -1431,29 +1805,28 @@ append_vertrans (struct ver_varinfo *var, unsigned long language,
/* Local functions used to write out an rc file. */
static void indent (FILE *, int);
-static void write_rc_directory
- (FILE *, const struct res_directory *, const struct res_id *,
- const struct res_id *, int *, int);
-static void write_rc_subdir
- (FILE *, const struct res_entry *, const struct res_id *,
- const struct res_id *, int *, int);
-static void write_rc_resource
- (FILE *, const struct res_id *, const struct res_id *,
- const struct res_resource *, int *);
-static void write_rc_accelerators (FILE *, const struct accelerator *);
-static void write_rc_cursor (FILE *, const struct cursor *);
-static void write_rc_group_cursor (FILE *, const struct group_cursor *);
-static void write_rc_dialog (FILE *, const struct dialog *);
-static void write_rc_dialog_control (FILE *, const struct dialog_control *);
-static void write_rc_fontdir (FILE *, const struct fontdir *);
-static void write_rc_group_icon (FILE *, const struct group_icon *);
-static void write_rc_menu (FILE *, const struct menu *, int);
-static void write_rc_menuitems (FILE *, const struct menuitem *, int, int);
-static void write_rc_rcdata (FILE *, const struct rcdata_item *, int);
-static void write_rc_stringtable
- (FILE *, const struct res_id *, const struct stringtable *);
-static void write_rc_versioninfo (FILE *, const struct versioninfo *);
-static void write_rc_filedata (FILE *, unsigned long, const unsigned char *);
+static void write_rc_directory (FILE *, const rc_res_directory *, const rc_res_id *,
+ const rc_res_id *, rc_uint_type *, int);
+static void write_rc_subdir (FILE *, const rc_res_entry *, const rc_res_id *,
+ const rc_res_id *, rc_uint_type *, int);
+static void write_rc_resource (FILE *, const rc_res_id *, const rc_res_id *,
+ const rc_res_resource *, rc_uint_type *);
+static void write_rc_accelerators (FILE *, const rc_accelerator *);
+static void write_rc_cursor (FILE *, const rc_cursor *);
+static void write_rc_group_cursor (FILE *, const rc_group_cursor *);
+static void write_rc_dialog (FILE *, const rc_dialog *);
+static void write_rc_dialog_control (FILE *, const rc_dialog_control *);
+static void write_rc_fontdir (FILE *, const rc_fontdir *);
+static void write_rc_group_icon (FILE *, const rc_group_icon *);
+static void write_rc_menu (FILE *, const rc_menu *, int);
+static void write_rc_toolbar (FILE *, const rc_toolbar *);
+static void write_rc_menuitems (FILE *, const rc_menuitem *, int, int);
+static void write_rc_messagetable (FILE *, rc_uint_type , const bfd_byte *);
+
+static void write_rc_datablock (FILE *, rc_uint_type , const bfd_byte *, int, int, int);
+static void write_rc_rcdata (FILE *, const rc_rcdata_item *, int);
+static void write_rc_stringtable (FILE *, const rc_res_id *, const rc_stringtable *);
+static void write_rc_versioninfo (FILE *, const rc_versioninfo *);
/* Indent a given number of spaces. */
@@ -1468,19 +1841,15 @@ indent (FILE *e, int c)
/* Dump the resources we have read in the format of an rc file.
- Actually, we don't use the format of an rc file, because it's way
- too much of a pain--for example, we'd have to write icon resources
- into a file and refer to that file. We just generate a readable
- format that kind of looks like an rc file, and is useful for
- understanding the contents of a resource file. Someday we may want
- to generate an rc file which the rc compiler can read; if that day
- comes, this code will have to be fixed up. */
+ Reasoned by the fact, that some resources need to be stored into file and
+ refer to that file, we use the user-data model for that to express it binary
+ without the need to store it somewhere externally. */
void
-write_rc_file (const char *filename, const struct res_directory *resources)
+write_rc_file (const char *filename, const rc_res_directory *resources)
{
FILE *e;
- int language;
+ rc_uint_type language;
if (filename == NULL)
e = stdout;
@@ -1491,9 +1860,9 @@ write_rc_file (const char *filename, const struct res_directory *resources)
fatal (_("can't open `%s' for output: %s"), filename, strerror (errno));
}
- language = -1;
- write_rc_directory (e, resources, (const struct res_id *) NULL,
- (const struct res_id *) NULL, &language, 1);
+ language = (rc_uint_type) ((bfd_signed_vma) -1);
+ write_rc_directory (e, resources, (const rc_res_id *) NULL,
+ (const rc_res_id *) NULL, &language, 1);
}
/* Write out a directory. E is the file to write to. RD is the
@@ -1503,20 +1872,23 @@ write_rc_file (const char *filename, const struct res_directory *resources)
language. LEVEL is the level in the tree. */
static void
-write_rc_directory (FILE *e, const struct res_directory *rd,
- const struct res_id *type, const struct res_id *name,
- int *language, int level)
+write_rc_directory (FILE *e, const rc_res_directory *rd,
+ const rc_res_id *type, const rc_res_id *name,
+ rc_uint_type *language, int level)
{
- const struct res_entry *re;
+ const rc_res_entry *re;
/* Print out some COFF information that rc files can't represent. */
-
+ if (rd->time != 0 || rd->characteristics != 0 || rd->major != 0 || rd->minor != 0)
+ {
+ wr_printcomment (e, "COFF information not part of RC");
if (rd->time != 0)
- fprintf (e, "// Time stamp: %lu\n", rd->time);
+ wr_printcomment (e, "Time stamp: %u", rd->time);
if (rd->characteristics != 0)
- fprintf (e, "// Characteristics: %lu\n", rd->characteristics);
+ wr_printcomment (e, "Characteristics: %u", rd->characteristics);
if (rd->major != 0 || rd->minor != 0)
- fprintf (e, "// Version: %d %d\n", rd->major, rd->minor);
+ wr_printcomment (e, "Version major:%d minor:%d", rd->major, rd->minor);
+ }
for (re = rd->entries; re != NULL; re = re->next)
{
@@ -1543,7 +1915,7 @@ write_rc_directory (FILE *e, const struct res_directory *rd,
&& re->id.u.id != (unsigned long) (unsigned int) *language
&& (re->id.u.id & 0xffff) == re->id.u.id)
{
- fprintf (e, "LANGUAGE %lu, %lu\n",
+ wr_print (e, "LANGUAGE %u, %u\n",
re->id.u.id & ((1 << SUBLANG_SHIFT) - 1),
(re->id.u.id >> SUBLANG_SHIFT) & 0xff);
*language = re->id.u.id;
@@ -1569,12 +1941,16 @@ write_rc_directory (FILE *e, const struct res_directory *rd,
}
else
{
- fprintf (e, "// Resource at unexpected level %d\n", level);
- write_rc_resource (e, type, (struct res_id *) NULL, re->u.res,
+ wr_printcomment (e, "Resource at unexpected level %d", level);
+ write_rc_resource (e, type, (rc_res_id *) NULL, re->u.res,
language);
}
}
}
+ if (rd->entries == NULL)
+ {
+ wr_print_flush (e);
+ }
}
/* Write out a subdirectory entry. E is the file to write to. RE is
@@ -1583,15 +1959,15 @@ write_rc_directory (FILE *e, const struct res_directory *rd,
LEVEL is the level in the tree. */
static void
-write_rc_subdir (FILE *e, const struct res_entry *re,
- const struct res_id *type, const struct res_id *name,
- int *language, int level)
+write_rc_subdir (FILE *e, const rc_res_entry *re,
+ const rc_res_id *type, const rc_res_id *name,
+ rc_uint_type *language, int level)
{
fprintf (e, "\n");
switch (level)
{
case 1:
- fprintf (e, "// Type: ");
+ wr_printcomment (e, "Type: ");
if (re->id.named)
res_id_print (e, re->id, 1);
else
@@ -1619,6 +1995,8 @@ write_rc_subdir (FILE *e, const struct res_entry *re,
case RT_VXD: s = "vxd"; break;
case RT_ANICURSOR: s = "anicursor"; break;
case RT_ANIICON: s = "aniicon"; break;
+ case RT_TOOLBAR: s = "toolbar"; break;
+ case RT_HTML: s = "html"; break;
default: s = NULL; break;
}
@@ -1627,25 +2005,21 @@ write_rc_subdir (FILE *e, const struct res_entry *re,
else
res_id_print (e, re->id, 1);
}
- fprintf (e, "\n");
break;
case 2:
- fprintf (e, "// Name: ");
+ wr_printcomment (e, "Name: ");
res_id_print (e, re->id, 1);
- fprintf (e, "\n");
break;
case 3:
- fprintf (e, "// Language: ");
+ wr_printcomment (e, "Language: ");
res_id_print (e, re->id, 1);
- fprintf (e, "\n");
break;
default:
- fprintf (e, "// Level %d: ", level);
+ wr_printcomment (e, "Level %d: ", level);
res_id_print (e, re->id, 1);
- fprintf (e, "\n");
}
write_rc_directory (e, re->u.dir, type, name, language, level + 1);
@@ -1658,38 +2032,36 @@ write_rc_subdir (FILE *e, const struct res_entry *re,
language. */
static void
-write_rc_resource (FILE *e, const struct res_id *type,
- const struct res_id *name, const struct res_resource *res,
- int *language)
+write_rc_resource (FILE *e, const rc_res_id *type,
+ const rc_res_id *name, const rc_res_resource *res,
+ rc_uint_type *language)
{
const char *s;
int rt;
int menuex = 0;
- fprintf (e, "\n");
-
switch (res->type)
{
default:
abort ();
case RES_TYPE_ACCELERATOR:
- s = "ACCELERATOR";
+ s = "ACCELERATORS";
rt = RT_ACCELERATOR;
break;
case RES_TYPE_BITMAP:
- s = "BITMAP";
+ s = "2 /* RT_BITMAP */";
rt = RT_BITMAP;
break;
case RES_TYPE_CURSOR:
- s = "CURSOR";
+ s = "1 /* RT_CURSOR */";
rt = RT_CURSOR;
break;
case RES_TYPE_GROUP_CURSOR:
- s = "GROUP_CURSOR";
+ s = "12 /* RT_GROUP_CURSOR */";
rt = RT_GROUP_CURSOR;
break;
@@ -1702,22 +2074,22 @@ write_rc_resource (FILE *e, const struct res_id *type,
break;
case RES_TYPE_FONT:
- s = "FONT";
+ s = "8 /* RT_FONT */";
rt = RT_FONT;
break;
case RES_TYPE_FONTDIR:
- s = "FONTDIR";
+ s = "7 /* RT_FONTDIR */";
rt = RT_FONTDIR;
break;
case RES_TYPE_ICON:
- s = "ICON";
+ s = "3 /* RT_ICON */";
rt = RT_ICON;
break;
case RES_TYPE_GROUP_ICON:
- s = "GROUP_ICON";
+ s = "14 /* RT_GROUP_ICON */";
rt = RT_GROUP_ICON;
break;
@@ -1736,7 +2108,7 @@ write_rc_resource (FILE *e, const struct res_id *type,
break;
case RES_TYPE_MESSAGETABLE:
- s = "MESSAGETABLE";
+ s = "11 /* RT_MESSAGETABLE */";
rt = RT_MESSAGETABLE;
break;
@@ -1759,32 +2131,76 @@ write_rc_resource (FILE *e, const struct res_id *type,
s = "VERSIONINFO";
rt = RT_VERSION;
break;
+
+ case RES_TYPE_TOOLBAR:
+ s = "TOOLBAR";
+ rt = RT_TOOLBAR;
+ break;
}
if (rt != 0
&& type != NULL
&& (type->named || type->u.id != (unsigned long) rt))
{
- fprintf (e, "// Unexpected resource type mismatch: ");
+ wr_printcomment (e, "Unexpected resource type mismatch: ");
res_id_print (e, *type, 1);
fprintf (e, " != %d", rt);
}
if (res->coff_info.codepage != 0)
- fprintf (e, "// Code page: %lu\n", res->coff_info.codepage);
+ wr_printcomment (e, "Code page: %u", res->coff_info.codepage);
if (res->coff_info.reserved != 0)
- fprintf (e, "// COFF reserved value: %lu\n", res->coff_info.reserved);
+ wr_printcomment (e, "COFF reserved value: %u", res->coff_info.reserved);
+ wr_print (e, "\n");
+ if (rt == RT_STRING)
+ ;
+ else
+ {
if (name != NULL)
- res_id_print (e, *name, 0);
+ res_id_print (e, *name, 1);
else
fprintf (e, "??Unknown-Name??");
-
fprintf (e, " ");
+ }
+
if (s != NULL)
fprintf (e, "%s", s);
else if (type != NULL)
+ {
+ if (type->named == 0)
+ {
+#define PRINT_RT_NAME(NAME) case NAME: \
+ fprintf (e, "%u /* %s */", (unsigned int) NAME, #NAME); \
+ break
+
+ switch (type->u.id)
+ {
+ default:
res_id_print (e, *type, 0);
+ break;
+
+ PRINT_RT_NAME(RT_MANIFEST);
+ PRINT_RT_NAME(RT_ANICURSOR);
+ PRINT_RT_NAME(RT_ANIICON);
+ PRINT_RT_NAME(RT_RCDATA);
+ PRINT_RT_NAME(RT_ICON);
+ PRINT_RT_NAME(RT_CURSOR);
+ PRINT_RT_NAME(RT_BITMAP);
+ PRINT_RT_NAME(RT_PLUGPLAY);
+ PRINT_RT_NAME(RT_VXD);
+ PRINT_RT_NAME(RT_FONT);
+ PRINT_RT_NAME(RT_FONTDIR);
+ PRINT_RT_NAME(RT_HTML);
+ PRINT_RT_NAME(RT_MESSAGETABLE);
+ PRINT_RT_NAME(RT_DLGINCLUDE);
+ PRINT_RT_NAME(RT_DLGINIT);
+ }
+#undef PRINT_RT_NAME
+ }
+ else
+ res_id_print (e, *type, 1);
+ }
else
fprintf (e, "??Unknown-Type??");
@@ -1802,11 +2218,17 @@ write_rc_resource (FILE *e, const struct res_id *type,
if (res->type == RES_TYPE_DIALOG)
{
- fprintf (e, " %d, %d, %d, %d", res->u.dialog->x, res->u.dialog->y,
- res->u.dialog->width, res->u.dialog->height);
+ fprintf (e, " %d, %d, %d, %d",
+ (int) res->u.dialog->x, (int) res->u.dialog->y,
+ (int) res->u.dialog->width, (int) res->u.dialog->height);
if (res->u.dialog->ex != NULL
&& res->u.dialog->ex->help != 0)
- fprintf (e, ", %lu", res->u.dialog->ex->help);
+ fprintf (e, ", %u", (unsigned int) res->u.dialog->ex->help);
+ }
+ else if (res->type == RES_TYPE_TOOLBAR)
+ {
+ fprintf (e, " %d, %d", (int) res->u.toolbar->button_width,
+ (int) res->u.toolbar->button_height);
}
fprintf (e, "\n");
@@ -1835,16 +2257,16 @@ write_rc_resource (FILE *e, const struct res_id *type,
if (res->res_info.language != 0 && res->res_info.language != *language)
fprintf (e, "%sLANGUAGE %d, %d\n",
modifiers ? "// " : "",
- res->res_info.language & ((1<<SUBLANG_SHIFT)-1),
- (res->res_info.language >> SUBLANG_SHIFT) & 0xff);
+ (int) res->res_info.language & ((1<<SUBLANG_SHIFT)-1),
+ (int) (res->res_info.language >> SUBLANG_SHIFT) & 0xff);
if (res->res_info.characteristics != 0)
- fprintf (e, "%sCHARACTERISTICS %lu\n",
+ fprintf (e, "%sCHARACTERISTICS %u\n",
modifiers ? "// " : "",
- res->res_info.characteristics);
+ (unsigned int) res->res_info.characteristics);
if (res->res_info.version != 0)
- fprintf (e, "%sVERSION %lu\n",
+ fprintf (e, "%sVERSION %u\n",
modifiers ? "// " : "",
- res->res_info.version);
+ (unsigned int) res->res_info.version);
}
switch (res->type)
@@ -1892,6 +2314,10 @@ write_rc_resource (FILE *e, const struct res_id *type,
write_rc_rcdata (e, res->u.userdata, 0);
break;
+ case RES_TYPE_TOOLBAR:
+ write_rc_toolbar (e, res->u.toolbar);
+ break;
+
case RES_TYPE_VERSIONINFO:
write_rc_versioninfo (e, res->u.versioninfo);
break;
@@ -1899,8 +2325,10 @@ write_rc_resource (FILE *e, const struct res_id *type,
case RES_TYPE_BITMAP:
case RES_TYPE_FONT:
case RES_TYPE_ICON:
+ write_rc_datablock (e, res->u.data.length, res->u.data.data, 0, 1, 0);
+ break;
case RES_TYPE_MESSAGETABLE:
- write_rc_filedata (e, res->u.data.length, res->u.data.data);
+ write_rc_messagetable (e, res->u.data.length, res->u.data.data);
break;
}
}
@@ -1908,9 +2336,9 @@ write_rc_resource (FILE *e, const struct res_id *type,
/* Write out accelerator information. */
static void
-write_rc_accelerators (FILE *e, const struct accelerator *accelerators)
+write_rc_accelerators (FILE *e, const rc_accelerator *accelerators)
{
- const struct accelerator *acc;
+ const rc_accelerator *acc;
fprintf (e, "BEGIN\n");
for (acc = accelerators; acc != NULL; acc = acc->next)
@@ -1923,16 +2351,16 @@ write_rc_accelerators (FILE *e, const struct accelerator *accelerators)
&& ISPRINT (acc->key)
&& (acc->flags & ACC_VIRTKEY) == 0)
{
- fprintf (e, "\"%c\"", acc->key);
+ fprintf (e, "\"%c\"", (char) acc->key);
printable = 1;
}
else
{
- fprintf (e, "%d", acc->key);
+ fprintf (e, "%d", (int) acc->key);
printable = 0;
}
- fprintf (e, ", %d", acc->id);
+ fprintf (e, ", %d", (int) acc->id);
if (! printable)
{
@@ -1959,41 +2387,60 @@ write_rc_accelerators (FILE *e, const struct accelerator *accelerators)
file, which the rc file would include. */
static void
-write_rc_cursor (FILE *e, const struct cursor *cursor)
+write_rc_cursor (FILE *e, const rc_cursor *cursor)
{
- fprintf (e, "// Hotspot: x: %d; y: %d\n", cursor->xhotspot,
- cursor->yhotspot);
- write_rc_filedata (e, cursor->length, cursor->data);
+ fprintf (e, "BEGIN\n");
+ indent (e, 2);
+ fprintf (e, " 0x%x, 0x%x,\t/* Hotspot x: %d, y: %d. */\n",
+ (unsigned int) cursor->xhotspot, (unsigned int) cursor->yhotspot,
+ (int) cursor->xhotspot, (int) cursor->yhotspot);
+ write_rc_datablock (e, (rc_uint_type) cursor->length, (const bfd_byte *) cursor->data,
+ 0, 0, 0);
+ fprintf (e, "END\n");
}
/* Write out group cursor data. This would normally be built from the
cursor data. */
static void
-write_rc_group_cursor (FILE *e, const struct group_cursor *group_cursor)
+write_rc_group_cursor (FILE *e, const rc_group_cursor *group_cursor)
{
- const struct group_cursor *gc;
+ const rc_group_cursor *gc;
+ int c;
+
+ for (c = 0, gc = group_cursor; gc != NULL; gc = gc->next, c++)
+ ;
+ fprintf (e, "BEGIN\n");
- for (gc = group_cursor; gc != NULL; gc = gc->next)
+ indent (e, 2);
+ fprintf (e, "0, 2, %d%s\t /* Having %d items. */\n", c, (c != 0 ? "," : ""), c);
+ indent (e, 4);
+ fprintf (e, "/* width, height, planes, bits, bytes, index. */\n");
+
+ for (c = 1, gc = group_cursor; gc != NULL; gc = gc->next, c++)
{
- fprintf (e, "// width: %d; height %d; planes %d; bits %d\n",
- gc->width, gc->height, gc->planes, gc->bits);
- fprintf (e, "// data bytes: %lu; index: %d\n",
- gc->bytes, gc->index);
+ indent (e, 4);
+ fprintf (e, "%d, %d, %d, %d, 0x%xL, %d%s /* Element %d. */\n",
+ (int) gc->width, (int) gc->height, (int) gc->planes, (int) gc->bits,
+ (unsigned int) gc->bytes, (int) gc->index, (gc->next != NULL ? "," : ""), c);
+ fprintf (e, "/* width: %d; height %d; planes %d; bits %d. */\n",
+ (int) gc->width, (int) gc->height, (int) gc->planes,
+ (int) gc->bits);
}
+ fprintf (e, "END\n");
}
/* Write dialog data. */
static void
-write_rc_dialog (FILE *e, const struct dialog *dialog)
+write_rc_dialog (FILE *e, const rc_dialog *dialog)
{
- const struct dialog_control *control;
+ const rc_dialog_control *control;
- fprintf (e, "STYLE 0x%lx\n", dialog->style);
+ fprintf (e, "STYLE 0x%x\n", dialog->style);
if (dialog->exstyle != 0)
- fprintf (e, "EXSTYLE 0x%lx\n", dialog->exstyle);
+ fprintf (e, "EXSTYLE 0x%x\n", (unsigned int) dialog->exstyle);
if ((dialog->class.named && dialog->class.u.n.length > 0)
|| dialog->class.u.id != 0)
@@ -2005,9 +2452,9 @@ write_rc_dialog (FILE *e, const struct dialog *dialog)
if (dialog->caption != NULL)
{
- fprintf (e, "CAPTION \"");
- unicode_print (e, dialog->caption, -1);
- fprintf (e, "\"\n");
+ fprintf (e, "CAPTION ");
+ unicode_print_quoted (e, dialog->caption, -1);
+ fprintf (e, "\n");
}
if ((dialog->menu.named && dialog->menu.u.n.length > 0)
@@ -2020,15 +2467,16 @@ write_rc_dialog (FILE *e, const struct dialog *dialog)
if (dialog->font != NULL)
{
- fprintf (e, "FONT %d, \"", dialog->pointsize);
- unicode_print (e, dialog->font, -1);
- fprintf (e, "\"");
+ fprintf (e, "FONT %d, ", (int) dialog->pointsize);
+ unicode_print_quoted (e, dialog->font, -1);
if (dialog->ex != NULL
&& (dialog->ex->weight != 0
|| dialog->ex->italic != 0
|| dialog->ex->charset != 1))
fprintf (e, ", %d, %d, %d",
- dialog->ex->weight, dialog->ex->italic, dialog->ex->charset);
+ (int) dialog->ex->weight,
+ (int) dialog->ex->italic,
+ (int) dialog->ex->charset);
fprintf (e, "\n");
}
@@ -2079,7 +2527,7 @@ static const struct control_info control_info[] =
/* Write a dialog control. */
static void
-write_rc_dialog_control (FILE *e, const struct dialog_control *control)
+write_rc_dialog_control (FILE *e, const rc_dialog_control *control)
{
const struct control_info *ci;
@@ -2100,7 +2548,10 @@ write_rc_dialog_control (FILE *e, const struct dialog_control *control)
else if (ci->name != NULL)
fprintf (e, "%s", ci->name);
else
+ {
fprintf (e, "CONTROL");
+ ci = NULL;
+ }
if (control->text.named || control->text.u.id != 0)
{
@@ -2109,7 +2560,7 @@ write_rc_dialog_control (FILE *e, const struct dialog_control *control)
fprintf (e, ",");
}
- fprintf (e, " %d, ", control->id);
+ fprintf (e, " %d, ", (int) control->id);
if (ci == NULL)
{
@@ -2118,10 +2569,10 @@ write_rc_dialog_control (FILE *e, const struct dialog_control *control)
res_id_print (e, control->class, 0);
if (control->class.named)
fprintf (e, "\"");
- fprintf (e, ", 0x%lx, ", control->style);
+ fprintf (e, ", 0x%x, ", (unsigned int) control->style);
}
- fprintf (e, "%d, %d", control->x, control->y);
+ fprintf (e, "%d, %d", (int) control->x, (int) control->y);
if (control->style != SS_ICON
|| control->exstyle != 0
@@ -2129,15 +2580,17 @@ write_rc_dialog_control (FILE *e, const struct dialog_control *control)
|| control->height != 0
|| control->help != 0)
{
- fprintf (e, ", %d, %d", control->width, control->height);
+ fprintf (e, ", %d, %d", (int) control->width, (int) control->height);
/* FIXME: We don't need to print the style if it is the default.
More importantly, in certain cases we actually need to turn
off parts of the forced style, by using NOT. */
- fprintf (e, ", 0x%lx", control->style);
+ if (ci != NULL)
+ fprintf (e, ", 0x%x", (unsigned int) control->style);
if (control->exstyle != 0 || control->help != 0)
- fprintf (e, ", 0x%lx, %lu", control->exstyle, control->help);
+ fprintf (e, ", 0x%x, %u", (unsigned int) control->exstyle,
+ (unsigned int) control->help);
}
fprintf (e, "\n");
@@ -2150,51 +2603,93 @@ write_rc_dialog_control (FILE *e, const struct dialog_control *control)
the font data. */
static void
-write_rc_fontdir (FILE *e, const struct fontdir *fontdir)
+write_rc_fontdir (FILE *e, const rc_fontdir *fontdir)
{
- const struct fontdir *fc;
+ const rc_fontdir *fc;
+ int c;
- for (fc = fontdir; fc != NULL; fc = fc->next)
+ for (c = 0, fc = fontdir; fc != NULL; fc = fc->next, c++)
+ ;
+ fprintf (e, "BEGIN\n");
+ indent (e, 2);
+ fprintf (e, "%d%s\t /* Has %d elements. */\n", c, (c != 0 ? "," : ""), c);
+ for (c = 1, fc = fontdir; fc != NULL; fc = fc->next, c++)
{
- fprintf (e, "// Font index: %d\n", fc->index);
- write_rc_filedata (e, fc->length, fc->data);
+ indent (e, 4);
+ fprintf (e, "%d,\t/* Font no %d with index %d. */\n",
+ (int) fc->index, c, (int) fc->index);
+ write_rc_datablock (e, (rc_uint_type) fc->length - 2,
+ (const bfd_byte *) fc->data + 4,fc->next != NULL,
+ 0, 0);
}
+ fprintf (e, "END\n");
}
/* Write out group icon data. This would normally be built from the
icon data. */
static void
-write_rc_group_icon (FILE *e, const struct group_icon *group_icon)
+write_rc_group_icon (FILE *e, const rc_group_icon *group_icon)
{
- const struct group_icon *gi;
+ const rc_group_icon *gi;
+ int c;
+
+ for (c = 0, gi = group_icon; gi != NULL; gi = gi->next, c++)
+ ;
- for (gi = group_icon; gi != NULL; gi = gi->next)
+ fprintf (e, "BEGIN\n");
+ indent (e, 2);
+ fprintf (e, " 0, 1, %d%s\t /* Has %d elements. */\n", c, (c != 0 ? "," : ""), c);
+
+ indent (e, 4);
+ fprintf (e, "/* \"width height colors pad\", planes, bits, bytes, index. */\n");
+ for (c = 1, gi = group_icon; gi != NULL; gi = gi->next, c++)
{
- fprintf (e, "// width: %d; height %d; colors: %d; planes %d; bits %d\n",
- gi->width, gi->height, gi->colors, gi->planes, gi->bits);
- fprintf (e, "// data bytes: %lu; index: %d\n",
- gi->bytes, gi->index);
+ indent (e, 4);
+ fprintf (e, "\"\\%03o\\%03o\\%03o\\%03o\", %d, %d, 0x%xL, %d%s\t/* Element no %d. */\n",
+ gi->width, gi->height, gi->colors, 0, (int) gi->planes, (int) gi->bits,
+ (unsigned int) gi->bytes, (int) gi->index, (gi->next != NULL ? "," : ""), c);
}
+ fprintf (e, "END\n");
}
/* Write out a menu resource. */
static void
-write_rc_menu (FILE *e, const struct menu *menu, int menuex)
+write_rc_menu (FILE *e, const rc_menu *menu, int menuex)
{
if (menu->help != 0)
- fprintf (e, "// Help ID: %lu\n", menu->help);
+ fprintf (e, "// Help ID: %u\n", (unsigned int) menu->help);
write_rc_menuitems (e, menu->items, menuex, 0);
}
+static void
+write_rc_toolbar (FILE *e, const rc_toolbar *tb)
+{
+ rc_toolbar_item *it;
+ indent (e, 0);
+ fprintf (e, "BEGIN\n");
+ it = tb->items;
+ while(it != NULL)
+ {
+ indent (e, 2);
+ if (it->id.u.id == 0)
+ fprintf (e, "SEPARATOR\n");
+ else
+ fprintf (e, "BUTTON %d\n", (int) it->id.u.id);
+ it = it->next;
+ }
+ indent (e, 0);
+ fprintf (e, "END\n");
+}
+
/* Write out menuitems. */
static void
-write_rc_menuitems (FILE *e, const struct menuitem *menuitems, int menuex,
+write_rc_menuitems (FILE *e, const rc_menuitem *menuitems, int menuex,
int ind)
{
- const struct menuitem *mi;
+ const rc_menuitem *mi;
indent (e, ind);
fprintf (e, "BEGIN\n");
@@ -2222,15 +2717,14 @@ write_rc_menuitems (FILE *e, const struct menuitem *menuitems, int menuex,
fprintf (e, " \"\"");
else
{
- fprintf (e, " \"");
- unicode_print (e, mi->text, -1);
- fprintf (e, "\"");
+ fprintf (e, " ");
+ unicode_print_quoted (e, mi->text, -1);
}
if (! menuex)
{
if (mi->popup == NULL)
- fprintf (e, ", %d", mi->id);
+ fprintf (e, ", %d", (int) mi->id);
if ((mi->type & MENUITEM_CHECKED) != 0)
fprintf (e, ", CHECKED");
@@ -2249,15 +2743,15 @@ write_rc_menuitems (FILE *e, const struct menuitem *menuitems, int menuex,
{
if (mi->id != 0 || mi->type != 0 || mi->state != 0 || mi->help != 0)
{
- fprintf (e, ", %d", mi->id);
+ fprintf (e, ", %d", (int) mi->id);
if (mi->type != 0 || mi->state != 0 || mi->help != 0)
{
- fprintf (e, ", %lu", mi->type);
+ fprintf (e, ", %u", (unsigned int) mi->type);
if (mi->state != 0 || mi->help != 0)
{
- fprintf (e, ", %lu", mi->state);
+ fprintf (e, ", %u", (unsigned int) mi->state);
if (mi->help != 0)
- fprintf (e, ", %lu", mi->help);
+ fprintf (e, ", %u", (unsigned int) mi->help);
}
}
}
@@ -2273,158 +2767,332 @@ write_rc_menuitems (FILE *e, const struct menuitem *menuitems, int menuex,
fprintf (e, "END\n");
}
-/* Write out an rcdata resource. This is also used for other types of
- resources that need to print arbitrary data. */
-
-static void
-write_rc_rcdata (FILE *e, const struct rcdata_item *rcdata, int ind)
+static int
+test_rc_datablock_unicode (rc_uint_type length, const bfd_byte *data)
{
- const struct rcdata_item *ri;
+ rc_uint_type i;
+ if ((length & 1) != 0)
+ return 0;
- indent (e, ind);
- fprintf (e, "BEGIN\n");
+ for (i = 0; i < length; i += 2)
+ {
+ if (data[i] == 0 && data[i + 1] == 0 && (i + 2) < length)
+ return 0;
+ if (data[i] == 0xff && data[i + 1] == 0xff)
+ return 0;
+ }
+ return 1;
+}
- for (ri = rcdata; ri != NULL; ri = ri->next)
+static int
+test_rc_datablock_text (rc_uint_type length, const bfd_byte *data)
+{
+ int has_nl;
+ rc_uint_type c;
+ rc_uint_type i;
+
+ if (length <= 1)
+ return 0;
+
+ has_nl = 0;
+ for (i = 0, c = 0; i < length; i++)
{
- if (ri->type == RCDATA_BUFFER && ri->u.buffer.length == 0)
- continue;
+ if (! ISPRINT (data[i]) && data[i] != '\n'
+ && ! (data[i] == '\r' && (i + 1) < length && data[i + 1] == '\n')
+ && data[i] != '\t'
+ && ! (data[i] == 0 && (i + 1) != length))
+ {
+ if (data[i] <= 7)
+ return 0;
+ c++;
+ }
+ else if (data[i] == '\n') has_nl++;
+ }
+ if (length > 80 && ! has_nl)
+ return 0;
+ c = (((c * 10000) + (i / 100) - 1)) / i;
+ if (c >= 150)
+ return 0;
+ return 1;
+}
- indent (e, ind + 2);
+static void
+write_rc_messagetable (FILE *e, rc_uint_type length, const bfd_byte *data)
+{
+ int has_error = 0;
+ const struct bin_messagetable *mt;
+ fprintf (e, "BEGIN\n");
- switch (ri->type)
- {
- default:
- abort ();
+ write_rc_datablock (e, length, data, 0, 0, 0);
- case RCDATA_WORD:
- fprintf (e, "%d", ri->u.word);
+ fprintf (e, "\n");
+ wr_printcomment (e, "MC syntax dump");
+ if (length < BIN_MESSAGETABLE_SIZE)
+ has_error = 1;
+ else
+ do {
+ rc_uint_type m, i;
+ mt = (const struct bin_messagetable *) data;
+ m = windres_get_32 (&wrtarget, mt->cblocks, length);
+ if (length < (BIN_MESSAGETABLE_SIZE + m * BIN_MESSAGETABLE_BLOCK_SIZE))
+ {
+ has_error = 1;
break;
+ }
+ for (i = 0; i < m; i++)
+ {
+ rc_uint_type low, high, offset;
+ const struct bin_messagetable_item *mti;
- case RCDATA_DWORD:
- fprintf (e, "%luL", ri->u.dword);
+ low = windres_get_32 (&wrtarget, mt->items[i].lowid, 4);
+ high = windres_get_32 (&wrtarget, mt->items[i].highid, 4);
+ offset = windres_get_32 (&wrtarget, mt->items[i].offset, 4);
+ while (low <= high)
+ {
+ rc_uint_type elen, flags;
+ if ((offset + BIN_MESSAGETABLE_ITEM_SIZE) > length)
+ {
+ has_error = 1;
break;
+ }
+ mti = (const struct bin_messagetable_item *) &data[offset];
+ elen = windres_get_16 (&wrtarget, mti->length, 2);
+ flags = windres_get_16 (&wrtarget, mti->flags, 2);
+ if ((offset + elen) > length)
+ {
+ has_error = 1;
+ break;
+ }
+ wr_printcomment (e, "MessageId = 0x%x", low);
+ wr_printcomment (e, "");
+ if ((flags & MESSAGE_RESOURCE_UNICODE) == MESSAGE_RESOURCE_UNICODE)
+ unicode_print (e, (const unichar *) mti->data,
+ (elen - BIN_MESSAGETABLE_ITEM_SIZE) / 2);
+ else
+ ascii_print (e, (const char *) mti->data,
+ (elen - BIN_MESSAGETABLE_ITEM_SIZE));
+ wr_printcomment (e,"");
+ ++low;
+ offset += elen;
+ }
+ }
+ } while (0);
+ if (has_error)
+ wr_printcomment (e, "Illegal data");
+ wr_print_flush (e);
+ fprintf (e, "END\n");
+}
- case RCDATA_STRING:
- {
- const char *s;
- unsigned long i;
+static void
+write_rc_datablock (FILE *e, rc_uint_type length, const bfd_byte *data, int has_next,
+ int hasblock, int show_comment)
+{
+ int plen;
+ if (hasblock)
+ fprintf (e, "BEGIN\n");
+
+ if (show_comment == -1)
+ {
+ if (test_rc_datablock_text(length, data))
+ {
+ rc_uint_type i, c;
+ for (i = 0; i < length;)
+ {
+ indent (e, 2);
+ fprintf (e, "\"");
+
+ for (c = 0; i < length && c < 160 && data[i] != '\n'; c++, i++)
+ ;
+ if (i < length && data[i] == '\n')
+ ++i, ++c;
+ ascii_print (e, (const char *) &data[i - c], c);
fprintf (e, "\"");
- s = ri->u.string.s;
- for (i = 0; i < ri->u.string.length; i++)
+ if (i < length)
+ fprintf (e, "\n");
+ }
+
+ if (i == 0)
{
- if (ISPRINT (*s))
- putc (*s, e);
- else
- fprintf (e, "\\%03o", *s);
+ indent (e, 2);
+ fprintf (e, "\"\"");
}
- fprintf (e, "\"");
- break;
+ if (has_next)
+ fprintf (e, ",");
+ fprintf (e, "\n");
+ if (hasblock)
+ fprintf (e, "END\n");
+ return;
}
+ if (test_rc_datablock_unicode (length, data))
+ {
+ rc_uint_type i, c;
+ for (i = 0; i < length;)
+ {
+ const unichar *u;
- case RCDATA_WSTRING:
+ u = (const unichar *) &data[i];
+ indent (e, 2);
fprintf (e, "L\"");
- unicode_print (e, ri->u.wstring.w, ri->u.wstring.length);
+
+ for (c = 0; i < length && c < 160 && u[c] != '\n'; c++, i += 2)
+ ;
+ if (i < length && u[c] == '\n')
+ i += 2, ++c;
+ unicode_print (e, u, c);
fprintf (e, "\"");
- break;
+ if (i < length)
+ fprintf (e, "\n");
+ }
- case RCDATA_BUFFER:
+ if (i == 0)
{
- unsigned long i;
- int first;
+ indent (e, 2);
+ fprintf (e, "L\"\"");
+ }
+ if (has_next)
+ fprintf (e, ",");
+ fprintf (e, "\n");
+ if (hasblock)
+ fprintf (e, "END\n");
+ return;
+ }
- /* Assume little endian data. */
+ show_comment = 0;
+ }
- first = 1;
- for (i = 0; i + 3 < ri->u.buffer.length; i += 4)
+ if (length != 0)
{
- unsigned long l;
- int j;
+ rc_uint_type i, max_row;
+ int first = 1;
- if (! first)
- indent (e, ind + 2);
- l = ((((((ri->u.buffer.data[i + 3] << 8)
- | ri->u.buffer.data[i + 2]) << 8)
- | ri->u.buffer.data[i + 1]) << 8)
- | ri->u.buffer.data[i]);
- fprintf (e, "%luL", l);
- if (i + 4 < ri->u.buffer.length || ri->next != NULL)
- fprintf (e, ",");
- for (j = 0; j < 4; ++j)
- if (! ISPRINT (ri->u.buffer.data[i + j])
- && ri->u.buffer.data[i + j] != 0)
- break;
- if (j >= 4)
+ max_row = (show_comment ? 4 : 8);
+ indent (e, 2);
+ for (i = 0; i + 3 < length;)
{
- fprintf (e, "\t// ");
- for (j = 0; j < 4; ++j)
+ rc_uint_type k;
+ rc_uint_type comment_start;
+
+ comment_start = i;
+
+ if (! first)
+ indent (e, 2);
+
+ for (k = 0; k < max_row && i + 3 < length; k++, i += 4)
{
- if (! ISPRINT (ri->u.buffer.data[i + j]))
- fprintf (e, "\\%03o", ri->u.buffer.data[i + j]);
+ if (k == 0)
+ plen = fprintf (e, "0x%lxL",
+ (long) windres_get_32 (&wrtarget, data + i, length - i));
else
+ plen = fprintf (e, " 0x%lxL",
+ (long) windres_get_32 (&wrtarget, data + i, length - i)) - 1;
+ if (has_next || (i + 4) < length)
{
- if (ri->u.buffer.data[i + j] == '\\')
- fprintf (e, "\\");
- fprintf (e, "%c", ri->u.buffer.data[i + j]);
+ if (plen>0 && plen < 11)
+ indent (e, 11 - plen);
+ fprintf (e, ",");
}
}
+ if (show_comment)
+ {
+ fprintf (e, "\t/* ");
+ ascii_print (e, (const char *) &data[comment_start], i - comment_start);
+ fprintf (e, ". */");
}
fprintf (e, "\n");
first = 0;
}
- if (i + 1 < ri->u.buffer.length)
+ if (i + 1 < length)
{
- int s;
- int j;
-
if (! first)
- indent (e, ind + 2);
- s = (ri->u.buffer.data[i + 1] << 8) | ri->u.buffer.data[i];
- fprintf (e, "%d", s);
- if (i + 2 < ri->u.buffer.length || ri->next != NULL)
- fprintf (e, ",");
- for (j = 0; j < 2; ++j)
- if (! ISPRINT (ri->u.buffer.data[i + j])
- && ri->u.buffer.data[i + j] != 0)
- break;
- if (j >= 2)
+ indent (e, 2);
+ plen = fprintf (e, "0x%x",
+ (int) windres_get_16 (&wrtarget, data + i, length - i));
+ if (has_next || i + 2 < length)
{
- fprintf (e, "\t// ");
- for (j = 0; j < 2; ++j)
- {
- if (! ISPRINT (ri->u.buffer.data[i + j]))
- fprintf (e, "\\%03o", ri->u.buffer.data[i + j]);
- else
- {
- if (ri->u.buffer.data[i + j] == '\\')
- fprintf (e, "\\");
- fprintf (e, "%c", ri->u.buffer.data[i + j]);
- }
+ if (plen > 0 && plen < 11)
+ indent (e, 11 - plen);
+ fprintf (e, ",");
}
+ if (show_comment)
+ {
+ fprintf (e, "\t/* ");
+ ascii_print (e, (const char *) &data[i], 2);
+ fprintf (e, ". */");
}
fprintf (e, "\n");
i += 2;
first = 0;
}
- if (i < ri->u.buffer.length)
+ if (i < length)
{
if (! first)
- indent (e, ind + 2);
- if ((ri->u.buffer.data[i] & 0x7f) == ri->u.buffer.data[i]
- && ISPRINT (ri->u.buffer.data[i]))
- fprintf (e, "\"%c\"", ri->u.buffer.data[i]);
- else
- fprintf (e, "\"\\%03o\"", ri->u.buffer.data[i]);
- if (ri->next != NULL)
+ indent (e, 2);
+ fprintf (e, "\"");
+ ascii_print (e, (const char *) &data[i], 1);
+ fprintf (e, "\"");
+ if (has_next)
fprintf (e, ",");
fprintf (e, "\n");
first = 0;
}
+ }
+ if (hasblock)
+ fprintf (e, "END\n");
+}
+
+/* Write out an rcdata resource. This is also used for other types of
+ resources that need to print arbitrary data. */
+
+static void
+write_rc_rcdata (FILE *e, const rc_rcdata_item *rcdata, int ind)
+{
+ const rc_rcdata_item *ri;
+
+ indent (e, ind);
+ fprintf (e, "BEGIN\n");
+
+ for (ri = rcdata; ri != NULL; ri = ri->next)
+ {
+ if (ri->type == RCDATA_BUFFER && ri->u.buffer.length == 0)
+ continue;
+ switch (ri->type)
+ {
+ default:
+ abort ();
+
+ case RCDATA_WORD:
+ indent (e, ind + 2);
+ fprintf (e, "%ld", (long) (ri->u.word & 0xffff));
+ break;
+
+ case RCDATA_DWORD:
+ indent (e, ind + 2);
+ fprintf (e, "%luL", (unsigned long) ri->u.dword);
+ break;
+
+ case RCDATA_STRING:
+ indent (e, ind + 2);
+ fprintf (e, "\"");
+ ascii_print (e, ri->u.string.s, ri->u.string.length);
+ fprintf (e, "\"");
+ break;
+
+ case RCDATA_WSTRING:
+ indent (e, ind + 2);
+ fprintf (e, "L\"");
+ unicode_print (e, ri->u.wstring.w, ri->u.wstring.length);
+ fprintf (e, "\"");
+ break;
+
+ case RCDATA_BUFFER:
+ write_rc_datablock (e, (rc_uint_type) ri->u.buffer.length,
+ (const bfd_byte *) ri->u.buffer.data,
+ ri->next != NULL, 0, -1);
break;
- }
}
if (ri->type != RCDATA_BUFFER)
@@ -2442,17 +3110,17 @@ write_rc_rcdata (FILE *e, const struct rcdata_item *rcdata, int ind)
/* Write out a stringtable resource. */
static void
-write_rc_stringtable (FILE *e, const struct res_id *name,
- const struct stringtable *stringtable)
+write_rc_stringtable (FILE *e, const rc_res_id *name,
+ const rc_stringtable *stringtable)
{
- unsigned long offset;
+ rc_uint_type offset;
int i;
if (name != NULL && ! name->named)
offset = (name->u.id - 1) << 4;
else
{
- fprintf (e, "// %s string table name\n",
+ fprintf (e, "/* %s string table name. */\n",
name == NULL ? "Missing" : "Invalid");
offset = 0;
}
@@ -2463,10 +3131,10 @@ write_rc_stringtable (FILE *e, const struct res_id *name,
{
if (stringtable->strings[i].length != 0)
{
- fprintf (e, " %lu, \"", offset + i);
- unicode_print (e, stringtable->strings[i].string,
+ fprintf (e, " %lu, ", (long) offset + i);
+ unicode_print_quoted (e, stringtable->strings[i].string,
stringtable->strings[i].length);
- fprintf (e, "\"\n");
+ fprintf (e, "\n");
}
}
@@ -2476,36 +3144,37 @@ write_rc_stringtable (FILE *e, const struct res_id *name,
/* Write out a versioninfo resource. */
static void
-write_rc_versioninfo (FILE *e, const struct versioninfo *versioninfo)
+write_rc_versioninfo (FILE *e, const rc_versioninfo *versioninfo)
{
- const struct fixed_versioninfo *f;
- const struct ver_info *vi;
+ const rc_fixed_versioninfo *f;
+ const rc_ver_info *vi;
f = versioninfo->fixed;
if (f->file_version_ms != 0 || f->file_version_ls != 0)
- fprintf (e, " FILEVERSION %lu, %lu, %lu, %lu\n",
- (f->file_version_ms >> 16) & 0xffff,
- f->file_version_ms & 0xffff,
- (f->file_version_ls >> 16) & 0xffff,
- f->file_version_ls & 0xffff);
+ fprintf (e, " FILEVERSION %u, %u, %u, %u\n",
+ (unsigned int) ((f->file_version_ms >> 16) & 0xffff),
+ (unsigned int) (f->file_version_ms & 0xffff),
+ (unsigned int) ((f->file_version_ls >> 16) & 0xffff),
+ (unsigned int) (f->file_version_ls & 0xffff));
if (f->product_version_ms != 0 || f->product_version_ls != 0)
- fprintf (e, " PRODUCTVERSION %lu, %lu, %lu, %lu\n",
- (f->product_version_ms >> 16) & 0xffff,
- f->product_version_ms & 0xffff,
- (f->product_version_ls >> 16) & 0xffff,
- f->product_version_ls & 0xffff);
+ fprintf (e, " PRODUCTVERSION %u, %u, %u, %u\n",
+ (unsigned int) ((f->product_version_ms >> 16) & 0xffff),
+ (unsigned int) (f->product_version_ms & 0xffff),
+ (unsigned int) ((f->product_version_ls >> 16) & 0xffff),
+ (unsigned int) (f->product_version_ls & 0xffff));
if (f->file_flags_mask != 0)
- fprintf (e, " FILEFLAGSMASK 0x%lx\n", f->file_flags_mask);
+ fprintf (e, " FILEFLAGSMASK 0x%x\n", (unsigned int) f->file_flags_mask);
if (f->file_flags != 0)
- fprintf (e, " FILEFLAGS 0x%lx\n", f->file_flags);
+ fprintf (e, " FILEFLAGS 0x%x\n", (unsigned int) f->file_flags);
if (f->file_os != 0)
- fprintf (e, " FILEOS 0x%lx\n", f->file_os);
+ fprintf (e, " FILEOS 0x%x\n", (unsigned int) f->file_os);
if (f->file_type != 0)
- fprintf (e, " FILETYPE 0x%lx\n", f->file_type);
+ fprintf (e, " FILETYPE 0x%x\n", (unsigned int) f->file_type);
if (f->file_subtype != 0)
- fprintf (e, " FILESUBTYPE 0x%lx\n", f->file_subtype);
+ fprintf (e, " FILESUBTYPE 0x%x\n", (unsigned int) f->file_subtype);
if (f->file_date_ms != 0 || f->file_date_ls != 0)
- fprintf (e, "// Date: %lu, %lu\n", f->file_date_ms, f->file_date_ls);
+ fprintf (e, "/* Date: %u, %u. */\n",
+ (unsigned int) f->file_date_ms, (unsigned int) f->file_date_ls);
fprintf (e, "BEGIN\n");
@@ -2515,22 +3184,22 @@ write_rc_versioninfo (FILE *e, const struct versioninfo *versioninfo)
{
case VERINFO_STRING:
{
- const struct ver_stringinfo *vs;
+ const rc_ver_stringinfo *vs;
fprintf (e, " BLOCK \"StringFileInfo\"\n");
fprintf (e, " BEGIN\n");
- fprintf (e, " BLOCK \"");
- unicode_print (e, vi->u.string.language, -1);
- fprintf (e, "\"\n");
+ fprintf (e, " BLOCK ");
+ unicode_print_quoted (e, vi->u.string.language, -1);
+ fprintf (e, "\n");
fprintf (e, " BEGIN\n");
for (vs = vi->u.string.strings; vs != NULL; vs = vs->next)
{
- fprintf (e, " VALUE \"");
- unicode_print (e, vs->key, -1);
- fprintf (e, "\", \"");
- unicode_print (e, vs->value, -1);
- fprintf (e, "\"\n");
+ fprintf (e, " VALUE ");
+ unicode_print_quoted (e, vs->key, -1);
+ fprintf (e, ", ");
+ unicode_print_quoted (e, vs->value, -1);
+ fprintf (e, "\n");
}
fprintf (e, " END\n");
@@ -2540,17 +3209,16 @@ write_rc_versioninfo (FILE *e, const struct versioninfo *versioninfo)
case VERINFO_VAR:
{
- const struct ver_varinfo *vv;
+ const rc_ver_varinfo *vv;
fprintf (e, " BLOCK \"VarFileInfo\"\n");
fprintf (e, " BEGIN\n");
- fprintf (e, " VALUE \"");
- unicode_print (e, vi->u.var.key, -1);
- fprintf (e, "\"");
+ fprintf (e, " VALUE ");
+ unicode_print_quoted (e, vi->u.var.key, -1);
for (vv = vi->u.var.var; vv != NULL; vv = vv->next)
fprintf (e, ", 0x%x, %d", (unsigned int) vv->language,
- vv->charset);
+ (int) vv->charset);
fprintf (e, "\n END\n");
@@ -2562,32 +3230,36 @@ write_rc_versioninfo (FILE *e, const struct versioninfo *versioninfo)
fprintf (e, "END\n");
}
-/* Write out data which would normally be read from a file. */
-
-static void
-write_rc_filedata (FILE *e, unsigned long length, const unsigned char *data)
+static rc_uint_type
+rcdata_copy (const rc_rcdata_item *src, bfd_byte *dst)
{
- unsigned long i;
-
- for (i = 0; i + 15 < length; i += 16)
- {
- fprintf (e, "// %4lx: ", i);
- fprintf (e, "%02x %02x %02x %02x %02x %02x %02x %02x ",
- data[i + 0], data[i + 1], data[i + 2], data[i + 3],
- data[i + 4], data[i + 5], data[i + 6], data[i + 7]);
- fprintf (e, "%02x %02x %02x %02x %02x %02x %02x %02x\n",
- data[i + 8], data[i + 9], data[i + 10], data[i + 11],
- data[i + 12], data[i + 13], data[i + 14], data[i + 15]);
- }
-
- if (i < length)
- {
- fprintf (e, "// %4lx:", i);
- while (i < length)
+ if (! src)
+ return 0;
+ switch (src->type)
{
- fprintf (e, " %02x", data[i]);
- ++i;
- }
- fprintf (e, "\n");
+ case RCDATA_WORD:
+ if (dst)
+ windres_put_16 (&wrtarget, dst, (rc_uint_type) src->u.word);
+ return 2;
+ case RCDATA_DWORD:
+ if (dst)
+ windres_put_32 (&wrtarget, dst, (rc_uint_type) src->u.dword);
+ return 4;
+ case RCDATA_STRING:
+ if (dst && src->u.string.length)
+ memcpy (dst, src->u.string.s, src->u.string.length);
+ return (rc_uint_type) src->u.string.length;
+ case RCDATA_WSTRING:
+ if (dst && src->u.wstring.length)
+ memcpy (dst, src->u.wstring.w, src->u.wstring.length * sizeof (unichar));
+ return (rc_uint_type) (src->u.wstring.length * sizeof (unichar));
+ case RCDATA_BUFFER:
+ if (dst && src->u.buffer.length)
+ memcpy (dst, src->u.buffer.data, src->u.buffer.length);
+ return (rc_uint_type) src->u.buffer.length;
+ default:
+ abort ();
}
+ /* Never reached. */
+ return 0;
}