mirror of
https://github.com/reactos/wine.git
synced 2024-12-01 07:30:37 +00:00
902da699bd
Fri Nov 3 20:08:17 1995 Alexandre Julliard <julliard@sunsite.unc.edu> * [configure.in] Attempt to check for -li386 on NetBSD. Please test this. Mon Oct 30 12:40:32 EST 1995 Jim Peterson <jspeter@birch.ee.vt.edu> * [*/*] Eliminated various warnings with either explicit casts or more accurate variable/parameter declarations (e.g. INT instead of short or WORD). Changed macros 'min' and 'max' to 'MIN' and 'MAX', since they're macros. * [controls/edit.c] [windows/defdlg.c] Added '#ifdef SUPERFLUOUS_FUNCTIONS' wrappers around function definition of EDIT_KeyVScrollDoc, EDIT_TextLineNumber, and DEFDLG_FindDefButton to avoid warnings. * [controls/button.c] [controls/scroll.c] [windows/defwnd.c] [windows/message.c] [windows/nonclient.c] Converted MAKEPOINT macro call to manual conversion. * [include/windows.h] For WINELIB32, structures POINT, SIZE, and RECT have LONG members instead of INT. This also invalidates the macro MAKEPOINT(), which is not supported in Win32. Also defined the POINTS structure (SHORT members) and the MAKEPOINTS macro. * [misc/commdlg.c] Changed a lot of 'strcpy' calls to 'strncpy' calls. I'm desperate to find this memory bug, and this should be done anyway. * [controls/edit.c] Well, the alteration mentioned above didn't do it, but #ifdef'ing out a few lines in EDIT_ClearText did. This leads to bugs, but for now, it's better than bizzare memory troubles. * [toolkit/miscstubs.c] Removed warning messages in GLOBAL_CreateBlock(), GLOBAL_FreeBlock(), and RELAY32_GetEntryPoint(). These are the most popular warnings, and their current implementation seems fine. Sat Oct 28 09:39:18 1995 Jochen Karrer <cip307@wpax01.Physik.Uni-Wuerzburg.DE> * [objects/cursoricon.c] Fix for "broken" X servers that invert masked cursor colors. Fri Oct 27 19:27:21 1995 Alex Korobka <alex@phm6.pharm.sunysb.edu> * [windows/dialog.c] [windows/nonclient.c] Remove unnecessary items from the system menu. Thu Oct 26 05:03:03 MET 1995 Philippe De Muyter <phdm@info.ucl.ac.be> * [objects/color.c] [objects/palette.c] Make GetNearestColor return a RGB value instead of a pixel value. Wed Oct 25 23:33:39 1995 Martin von Loewis <loewis@informatik.hu-berlin.de> * [controls/desktop.c][controls/menu.c][include/menu.h] Changed WORD parameters to UINT parameters. * [include/wintypes.h] Made UINT 32bit for WINELIB. * [loader/main.c] Disabled RELAY32_Init and MODULE_Init for WINELIB. * [misc/main.c] Assume CPU386 for WINELIB. * [rc/winerc] add_popup: set MF_POPUP flag on menu item. * [toolkit/Makefile.in][toolkit/hello3.c][toolkit/hello3res.c] Add resource demo hello3 for WINELIB. New file README.resources. * [toolkit/miscstubs.c] Add a case for 17. Tue Oct 17 15:13:10 IST 1995 Itai Nahshon <nahshon@vnet.ibm.com> * [loader/module.c] Do not append .exe if the file name already has an extension. * [misc/profile.c] Avoid creating a file with a junk name if a .ini file does not exist. * [if1632/gdi.spec] [if1632/user.spec] [if1632/dummy.c] Added a lot of dummy stubs for Windows Hebrew version. Tue Oct 17 01:03:24 1995 William Magro <wmagro@tc.cornell.edu> * [controls/button.c] Fix for buttons with no label. * [controls/combo.c][controls/listbox.c] Fixes for scrollbar positioning. Now disappears correctly for short lists. * [controls/edit.c] Handle memory allocation differently when building as library. * [controls/static] Don't destroy old icon before drawing new icon. (Fixes landscape/ portrait toggle icon in print dialog.) * [if1632/gdi.spec] New functions SetMetaFileBits and GetMetaFileBits * [include/sysmetrics.h] [windows/sysmetrics.c] Add support for GetSystemMetrics(SM_CMETRICS) == SM_CMETRICS * [include/windows.h] META_EXTTEXTOUT, not META_SETTEXTOUT define GetCurrentTime as GetTickCount (for wine library) * [loader/main.c] Don't initialize built-in modules in wine library * [memory/local.c] LocalReAlloc was defined incorrectly. Swap flags and size arguments. * [misc/main.c] Always report CPUTYPE=4 to wine library. * [objects/dib.c] RLE8 images were missing top line when decompressed. * [objects/metafile.c] SetMetaFileBits and GetMetaFileBits implemented. Works when called from winhelp. More testing needed. Various memory leaks plugged. Various other bug fixes. New metafile operations added in PlayMetaFileRecord: CreatePalette, SetTextAlign, SelectPalette, SetMapperFlags, RealizePalette, ExtTextOut, Escape. Testing needed. * [toolkit/heap.c] LocalUnLock changed to LocalUnlock Sun Oct 15 21:55:33 1995 Anand Kumria <akumria@ozemail.com.au> * [misc/winsock.c] Return the correct error number, for host lookup operations. Also, correct the problem with send_message. Fri Oct 13 19:04:35 1995 Morten Welinder <terra@diku.dk> * [Makefile.in] Using nm's built-in sorting. * [*/*.c] Use xmalloc for malloc and xrealloc for realloc in all ungarded cases. * [debugger/dbg.y] Handle C-like expressions. Clean-up. * [debugger/debug.l] Lots of new tokens for expressions. * [debugger/info.c] Implement "list" command for disassembling. * [misc/ole2nls.c] Implement more Danish stuff. Fri Oct 6 10:39:39 1995 Ram'on Garc'ia <ramon@ie3.clubs.etsit.upm.es> * [loader/module.c] Updated self-loading modules to support for new 32 bit stack frames.
669 lines
16 KiB
C
669 lines
16 KiB
C
/*
|
|
*
|
|
* Copyright Martin von Loewis, 1994
|
|
*
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/fcntl.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <windows.h>
|
|
#include <neexe.h>
|
|
#include "parser.h"
|
|
#include "y.tab.h"
|
|
|
|
char usage[]="winerc -bdvc -p prefix -o outfile < infile \n"
|
|
" -b Create a C array from a binary .res file\n"
|
|
" -d Output debugging information\n"
|
|
" -p prefix Give a prefix for the generated names\n"
|
|
" -v Show each resource as it is processed\n"
|
|
" -o file Output to file.c and file.h\n";
|
|
|
|
|
|
/*might be overwritten by command line*/
|
|
char *prefix="_Resource";
|
|
int verbose,constant;
|
|
gen_res* g_start;
|
|
FILE *header,*code;
|
|
char hname[256],sname[256];
|
|
|
|
int transform_binary_file(void);
|
|
int yyparse(void);
|
|
|
|
static void *xmalloc (size_t size)
|
|
{
|
|
void *res;
|
|
|
|
res = malloc (size ? size : 1);
|
|
if (res == NULL)
|
|
{
|
|
fprintf (stderr, "Virtual memory exhausted.\n");
|
|
exit (1);
|
|
}
|
|
return res;
|
|
}
|
|
|
|
|
|
int main(int argc,char *argv[])
|
|
{
|
|
extern int yydebug;
|
|
extern char* optarg;
|
|
int optc,lose,ret,binary;
|
|
lose=binary=0;
|
|
while((optc=getopt(argc,argv,"bdp:vo:"))!=EOF)
|
|
switch(optc)
|
|
{
|
|
/* bison will print state transitions on stderr */
|
|
case 'b':binary=1;
|
|
break;
|
|
case 'd':yydebug=1;
|
|
setbuf(stdout,0);
|
|
setbuf(stderr,0);
|
|
break;
|
|
case 'p':prefix=optarg;break;
|
|
case 'c':constant=1;break;
|
|
case 'v':verbose=1;
|
|
setbuf(stderr,0);
|
|
break;
|
|
case 'o':set_out_file(optarg);break;
|
|
default: lose++;break;
|
|
}
|
|
if(lose)return fprintf(stderr,usage),1;
|
|
if(!header)header=stdout;
|
|
if(!code)code=stdout;
|
|
if(binary)
|
|
ret=transform_binary_file();
|
|
else
|
|
ret=yyparse();
|
|
fclose(header);
|
|
fclose(code);
|
|
return ret;
|
|
}
|
|
|
|
void set_out_file(char *prefix)
|
|
{
|
|
sprintf(sname,"%s.c",prefix);
|
|
code=fopen(sname,"w");
|
|
sprintf(hname,"%s.h",prefix);
|
|
header=fopen(hname,"w");
|
|
}
|
|
|
|
int transform_binary_file()
|
|
{
|
|
int i,c;
|
|
fprintf(header,"#define APPLICATION_HAS_RESOURCES 1\n");
|
|
fprintf(code,"char _Application_resources[]={");
|
|
for(i=0;;i++)
|
|
{
|
|
c=getchar();
|
|
if(c==-1)break;
|
|
if(i%16==0)fputc('\n',code);
|
|
fprintf(code,"%3d,",c);
|
|
}
|
|
fprintf(code,"\n0}\nint _Aplication_resources_size=%d;\n",i);
|
|
return 0;
|
|
}
|
|
|
|
/* SunOS' memcpy is wrong for overlapping arrays */
|
|
char *save_memcpy(char *d,char* s,int l)
|
|
{
|
|
if(d<s)
|
|
for(;l;l--)*d++=*s++;
|
|
else
|
|
for(d+=l-1,s+=l-1;l;l--)*d--=*s--;
|
|
return d;
|
|
}
|
|
|
|
/*allow unaligned access*/
|
|
void put_WORD(unsigned char* p,WORD w)
|
|
{
|
|
*p=w&0xFF;
|
|
*(p+1)=w>>8;
|
|
}
|
|
|
|
void put_DWORD(unsigned char* p,DWORD d)
|
|
{
|
|
put_WORD(p,d&0xFFFF);
|
|
put_WORD(p+2,d>>16);
|
|
}
|
|
|
|
WORD get_WORD(unsigned char* p)
|
|
{
|
|
return *p|(*(p+1)<<8);
|
|
}
|
|
|
|
DWORD get_DWORD(unsigned char* p)
|
|
{
|
|
return get_WORD(p)|(get_WORD(p+2)<<16);
|
|
}
|
|
|
|
|
|
/*create a new gen_res, initial size 100*/
|
|
gen_res *new_res()
|
|
{ gen_res* ret=xmalloc(sizeof(gen_res)+100);
|
|
int i;
|
|
for(i=0;i<sizeof(gen_res)+100;i++)*((char*)ret+i)='\0';
|
|
ret->g_next=g_start;
|
|
ret->g_prev=0;
|
|
g_start=ret;
|
|
ret->space=100;
|
|
return ret;
|
|
}
|
|
|
|
/*double the space*/
|
|
gen_res* grow(gen_res* res)
|
|
{
|
|
res=realloc(res,sizeof(gen_res)+2*res->space);
|
|
if(!res)
|
|
fprintf(stderr,"Out of memory\n"),exit(1);
|
|
if(!res->g_prev)g_start=res;
|
|
else res->g_prev->g_next=res;
|
|
if(res->g_next)res->g_next->g_prev=res;
|
|
res->space=2*res->space;
|
|
return res;
|
|
}
|
|
|
|
|
|
/* insert bytes at offset 0, increase num_entries */
|
|
gen_res* insert_at_beginning(gen_res* res,char* entry,int size)
|
|
{
|
|
while(res->size+size>res->space)res=grow(res);
|
|
save_memcpy(res->res+size,res->res,res->size);
|
|
save_memcpy(res->res,entry,size);
|
|
res->size+=size;
|
|
res->num_entries++;
|
|
return res;
|
|
}
|
|
|
|
/* insert length characters from bytes into res, starting at start */
|
|
gen_res* insert_bytes(gen_res* res,char* bytes,int start,int length)
|
|
{
|
|
while(res->size+length>res->space)res=grow(res);
|
|
save_memcpy(res->res+start+length,res->res+start,res->size-start);
|
|
save_memcpy(res->res+start,bytes,length);
|
|
res->size+=length;
|
|
return res;
|
|
}
|
|
|
|
/*delete len bytes from res, starting at start*/
|
|
gen_res* delete_bytes(gen_res* res,int start,int len)
|
|
{
|
|
save_memcpy(res->res+start,res->res+start+len,res->size-start-len);
|
|
res->size-=len;
|
|
return res;
|
|
}
|
|
|
|
/*create a new style*/
|
|
rc_style *new_style()
|
|
{
|
|
rc_style *ret=xmalloc(sizeof(rc_style));
|
|
/*initially, no bits have to be reset*/
|
|
ret->and=-1;
|
|
/*initially, no bits are set*/
|
|
ret->or=0;
|
|
return ret;
|
|
}
|
|
|
|
/* entries are inserted at the beginning, starting from the last one */
|
|
gen_res* add_accelerator(int ev, int id, int flags, gen_res* prev)
|
|
{
|
|
char accel_entry[5];
|
|
if(prev->num_entries==0)flags|=0x80; /* last entry */
|
|
accel_entry[0]=flags;
|
|
put_WORD(accel_entry+1,ev);
|
|
put_WORD(accel_entry+3,id);
|
|
return insert_at_beginning(prev,accel_entry,5);
|
|
}
|
|
|
|
|
|
/* create an integer from the event, taking things as "^c" into account
|
|
add this as new entry */
|
|
gen_res* add_string_accelerator(char *ev, int id, int flags, gen_res* prev)
|
|
{
|
|
int event;
|
|
if(*ev=='^')
|
|
event=ev[1]-'a';
|
|
else
|
|
event=ev[0];
|
|
return add_accelerator(event,id,flags,prev);
|
|
}
|
|
|
|
/*is there a difference between ASCII and VIRTKEY accelerators? */
|
|
|
|
gen_res* add_ascii_accelerator(int ev, int id, int flags, gen_res* prev)
|
|
{
|
|
return add_accelerator(ev,id,flags,prev);
|
|
}
|
|
|
|
gen_res* add_vk_accelerator(int ev, int id, int flags, gen_res* prev)
|
|
{
|
|
return add_accelerator(ev,id,flags,prev);
|
|
}
|
|
|
|
/* create a new dialog header, set all items to 0 */
|
|
gen_res* new_dialog()
|
|
{ gen_res* ret=new_res();
|
|
ret->size=16; /*all strings "\0", no font*/
|
|
return ret;
|
|
}
|
|
|
|
/* the STYLE option was specified */
|
|
gen_res* dialog_style(rc_style* style, gen_res* attr)
|
|
{
|
|
/* default dialog styles? Do we need style->and? */
|
|
/* DS_SETFONT might have been specified before */
|
|
put_DWORD(attr->res,get_DWORD(attr->res)|style->or);
|
|
return attr;
|
|
}
|
|
|
|
/* menu name is at offset 13 */
|
|
int dialog_get_menu(gen_res* attr)
|
|
{
|
|
return 13;
|
|
}
|
|
|
|
/* the class is after the menu name */
|
|
int dialog_get_class(gen_res* attr)
|
|
{
|
|
int offs=dialog_get_menu(attr);
|
|
while(attr->res[offs])offs++;
|
|
offs++;
|
|
return offs;
|
|
}
|
|
|
|
/* the caption is after the class */
|
|
int dialog_get_caption(gen_res* attr)
|
|
{
|
|
int offs=dialog_get_class(attr);
|
|
while(attr->res[offs])offs++;
|
|
offs++;
|
|
return offs;
|
|
}
|
|
|
|
/* the fontsize, if present, is after the caption, followed by the font name */
|
|
int dialog_get_fontsize(gen_res* attr)
|
|
{
|
|
int offs=dialog_get_caption(attr);
|
|
while(attr->res[offs])offs++;
|
|
offs++;
|
|
return offs;
|
|
}
|
|
|
|
|
|
/* the CAPTION option was specified */
|
|
gen_res* dialog_caption(char* cap, gen_res*attr)
|
|
{
|
|
/* we don't need the terminating 0 as it's already there */
|
|
return insert_bytes(attr,cap,dialog_get_caption(attr),strlen(cap));
|
|
}
|
|
|
|
|
|
/* the FONT option was specified, set the DS_SETFONT flag */
|
|
gen_res* dialog_font(short size,char* font,gen_res *attr)
|
|
{
|
|
char c_size[2];
|
|
int offs=dialog_get_fontsize(attr);
|
|
put_DWORD(attr->res,get_DWORD(attr->res)|DS_SETFONT);
|
|
put_WORD(c_size,size);
|
|
attr=insert_bytes(attr,c_size,offs,2);
|
|
offs+=2;
|
|
/* as there is no font name by default, copy the '\0' */
|
|
return insert_bytes(attr,font,offs,strlen(font)+1);
|
|
}
|
|
|
|
gen_res* dialog_class(char* cap, gen_res*attr)
|
|
{
|
|
return insert_bytes(attr,cap,dialog_get_class(attr),strlen(cap));
|
|
}
|
|
|
|
gen_res* dialog_menu(char* cap, gen_res*attr)
|
|
{
|
|
return insert_bytes(attr,cap,dialog_get_menu(attr),strlen(cap));
|
|
}
|
|
|
|
/* set the dialogs id, position, extent, and style */
|
|
gen_res* create_control_desc(int id,int x,int y,int cx, int cy,rc_style *style)
|
|
{ gen_res* ret=new_res();
|
|
int s=WS_VISIBLE|WS_CHILD; /*defaults styles for any control*/
|
|
put_WORD(ret->res+0,x);
|
|
put_WORD(ret->res+2,y);
|
|
put_WORD(ret->res+4,cx);
|
|
put_WORD(ret->res+6,cy);
|
|
put_WORD(ret->res+8,id);
|
|
if(style)s=(s|style->or)&style->and;
|
|
put_DWORD(ret->res+10,s);
|
|
ret->size=17; /*empty strings, unused byte*/
|
|
return ret;
|
|
}
|
|
|
|
/* insert the control's label */
|
|
gen_res* label_control_desc(char* label,gen_res* cd)
|
|
{
|
|
int offs;
|
|
if(cd->res[14]&0x80)offs=15; /* one-character class */
|
|
else {
|
|
for(offs=14;cd->res[offs];offs++);
|
|
offs++;
|
|
}
|
|
return insert_bytes(cd,label,offs,strlen(label));
|
|
}
|
|
|
|
/* a CONTROL was specified */
|
|
gen_res* create_generic_control(char* label,int id,char* class,
|
|
rc_style*style,int x,int y,int cx,int cy)
|
|
{ char cl;
|
|
gen_res* ret=new_res();
|
|
put_WORD(ret->res+0,x);
|
|
put_WORD(ret->res+2,y);
|
|
put_WORD(ret->res+4,cx);
|
|
put_WORD(ret->res+6,cy);
|
|
put_WORD(ret->res+8,id);
|
|
put_DWORD(ret->res+10,style->or);
|
|
ret->size=17;
|
|
ret=insert_bytes(ret,label,15,strlen(label));
|
|
/* is it a predefined class? */
|
|
cl=0;
|
|
if(!strcmp(class,"BUTTON"))cl=CT_BUTTON;
|
|
if(!strcmp(class,"EDIT"))cl=CT_EDIT;
|
|
if(!strcmp(class,"STATIC"))cl=CT_STATIC;
|
|
if(!strcmp(class,"LISTBOX"))cl=CT_LISTBOX;
|
|
if(!strcmp(class,"SCROLLBAR"))cl=CT_SCROLLBAR;
|
|
if(!strcmp(class,"COMBOBOX"))cl=CT_COMBOBOX;
|
|
if(cl)ret->res[14]=cl;
|
|
else ret=insert_bytes(ret,class,14,strlen(class));
|
|
return ret;
|
|
}
|
|
|
|
/* insert cd into rest, set the type, add flags */
|
|
gen_res* add_control(int type,int flags,gen_res*cd,gen_res* rest)
|
|
{
|
|
put_DWORD(cd->res+10,get_DWORD(cd->res+10)|flags);
|
|
cd->res[14]=type;
|
|
return insert_at_beginning(rest,cd->res,cd->size);
|
|
}
|
|
|
|
/* an ICON control was specified, whf contains width, height, and flags */
|
|
gen_res* add_icon(char* name,int id,int x,int y,gen_res* whf,gen_res* rest)
|
|
{
|
|
put_WORD(whf->res+0,x);
|
|
put_WORD(whf->res+2,y);
|
|
put_WORD(whf->res+8,id);
|
|
whf=label_control_desc(name,whf);
|
|
return add_control(CT_STATIC,SS_ICON,whf,rest);
|
|
}
|
|
|
|
/* insert the generic control into rest */
|
|
gen_res* add_generic_control(gen_res* ctl, gen_res* rest)
|
|
{
|
|
return insert_at_beginning(rest,ctl->res,ctl->size);
|
|
}
|
|
|
|
/* create a dialog resource by inserting the header into the controls.
|
|
Set position and extent */
|
|
gen_res* make_dialog(gen_res* header,int x,int y,int cx,int cy,gen_res* ctls)
|
|
{
|
|
header->res[4]=ctls->num_entries;
|
|
header->type=dlg;
|
|
put_WORD(header->res+5,x);
|
|
put_WORD(header->res+7,y);
|
|
put_WORD(header->res+9,cx);
|
|
put_WORD(header->res+11,cy);
|
|
return insert_bytes(header,ctls->res,header->size,ctls->size);
|
|
}
|
|
|
|
/* create {0x15,0x16,0xFF} from '15 16 FF' */
|
|
gen_res *hex_to_raw(char *hex, gen_res*rest)
|
|
{
|
|
char r2[16];
|
|
int i;
|
|
for(i=0;*hex!='\'';i++)r2[i]=strtoul(hex,&hex,16);
|
|
return insert_bytes(rest,r2,0,i);
|
|
}
|
|
|
|
/* create a bitmap resource */
|
|
gen_res *make_bitmap(gen_res* res)
|
|
{
|
|
res=delete_bytes(res,0,14); /* skip bitmap file header*/
|
|
res->type=bmp;
|
|
return res;
|
|
}
|
|
|
|
gen_res *make_icon(gen_res* res)
|
|
{
|
|
res->type=ico;
|
|
return res;
|
|
}
|
|
|
|
gen_res *make_cursor(gen_res* res)
|
|
{
|
|
res->type=cur;
|
|
return res;
|
|
}
|
|
|
|
/* load resource bytes from the file name */
|
|
gen_res *load_file(char* name)
|
|
{
|
|
gen_res *res;
|
|
struct stat st;
|
|
int f=open(name,O_RDONLY);
|
|
if(!f)perror(name);
|
|
fstat(f,&st);
|
|
res=new_res();
|
|
while(res->space<st.st_size)res=grow(res);
|
|
read(f,res->res,st.st_size);
|
|
res->size=st.st_size;
|
|
close(f);
|
|
return res;
|
|
}
|
|
|
|
/* insert a normal menu item into res, starting from the last item */
|
|
gen_res *add_menuitem(char* name,int id,int flags,gen_res *res)
|
|
{
|
|
char item[4];
|
|
if(res->num_entries==0)flags|=MF_END;
|
|
put_WORD(item,flags);
|
|
put_WORD(item+2,id);
|
|
res=insert_at_beginning(res,name,strlen(name)+1);
|
|
res=insert_bytes(res,item,0,4);
|
|
return res;
|
|
}
|
|
|
|
/* insert a popup item into res */
|
|
gen_res *add_popup(char *name,short flags, gen_res* body, gen_res*res)
|
|
{
|
|
char c_flags[2];
|
|
flags|=MF_POPUP;
|
|
if(res->num_entries==0)flags|=MF_END;
|
|
put_WORD(c_flags,flags);
|
|
res=insert_at_beginning(res,body->res,body->size);
|
|
res=insert_bytes(res,name,0,strlen(name)+1);
|
|
res=insert_bytes(res,c_flags,0,2);
|
|
return res;
|
|
}
|
|
|
|
/* prefix the menu header into res */
|
|
gen_res *make_menu(gen_res* res)
|
|
{
|
|
static char header[4]={0,0,0,0};
|
|
res=insert_at_beginning(res,header,4);
|
|
res->type=men;
|
|
return res;
|
|
}
|
|
|
|
/* link top-level resources */
|
|
gen_res *add_resource(gen_res* first,gen_res *rest)
|
|
{
|
|
first->next=rest;
|
|
return first;
|
|
}
|
|
|
|
char *get_typename(gen_res* t)
|
|
{
|
|
switch(t->type){
|
|
case acc:return "ACCELERATOR";
|
|
case bmp:return "BITMAP";
|
|
case cur:return "CURSOR";
|
|
case dlg:return "DIALOG";
|
|
case fnt:return "FONT";
|
|
case ico:return "ICON";
|
|
case men:return "MENU";
|
|
case rdt:return "RCDATA";
|
|
case str:return "STRINGTABLE";
|
|
default: return "UNKNOWN";
|
|
}
|
|
}
|
|
|
|
/* create strings like _Sysres_DIALOG_2 */
|
|
char *get_resource_name(gen_res*it)
|
|
{
|
|
static char buf[1000];
|
|
if(it->n_type)
|
|
sprintf(buf,"%s_%s_%s",prefix,get_typename(it),it->n.s_name);
|
|
else
|
|
sprintf(buf,"%s_%s_%d",prefix,get_typename(it),it->n.i_name);
|
|
return buf;
|
|
}
|
|
|
|
#define ISCONSTANT (constant ? "const " : "")
|
|
|
|
/* create the final output */
|
|
void create_output(gen_res* top)
|
|
{
|
|
gen_res *it;
|
|
|
|
|
|
fprintf( header, "/* %s\n"
|
|
" * This file is automatically generated. Do not edit!\n"
|
|
" */\n\n"
|
|
"struct resource\n"
|
|
"{\n"
|
|
" int id, type;\n"
|
|
" char *name;\n"
|
|
" unsigned char *bytes;\n"
|
|
" unsigned int size;\n"
|
|
"};\n\n", hname );
|
|
|
|
/* Declare the resources */
|
|
|
|
for (it=top;it;it=it->next)
|
|
fprintf( header,"extern %sstruct resource %s;\n",
|
|
ISCONSTANT, get_resource_name(it) );
|
|
fprintf( header,"\nextern %sstruct resource * %sTable[];\n",
|
|
ISCONSTANT, prefix );
|
|
|
|
/* Print the resources bytes */
|
|
|
|
fprintf( code, "/* %s\n"
|
|
" * This file is automatically generated. Do not edit!\n"
|
|
" */\n\n"
|
|
"#include \"%s\"\n", sname, hname );
|
|
for(it=top;it;it=it->next)
|
|
{
|
|
int i;
|
|
fprintf( code, "static %sunsigned char %s__bytes[] = {\n",
|
|
ISCONSTANT, get_resource_name(it) );
|
|
for (i=0;i<it->size-1;i++)
|
|
{
|
|
fprintf(code,"%#4x,",it->res[i]);
|
|
if ((i&7)==7)fputc('\n',code);
|
|
}
|
|
fprintf(code,"%#4x};\n\n",it->res[i]);
|
|
}
|
|
|
|
/* Print the resources */
|
|
for (it=top;it;it=it->next)
|
|
{
|
|
int type;
|
|
switch(it->type)
|
|
{
|
|
case acc:type=NE_RSCTYPE_ACCELERATOR;break;
|
|
case bmp:type=NE_RSCTYPE_BITMAP;break;
|
|
case cur:type=NE_RSCTYPE_CURSOR;break;
|
|
case dlg:type=NE_RSCTYPE_DIALOG;break;
|
|
case fnt:type=NE_RSCTYPE_FONT;break;
|
|
case ico:type=NE_RSCTYPE_ICON;break;
|
|
case men:type=NE_RSCTYPE_MENU;break;
|
|
case rdt:type=NE_RSCTYPE_RCDATA;break;
|
|
case str:type=NE_RSCTYPE_STRING;break;
|
|
default:fprintf(stderr,"Unknown restype\n");type=-1;break;
|
|
}
|
|
if(it->n_type)
|
|
fprintf(code,"%sstruct resource %s = {0,%d,\"%s\",%s__bytes,%d};\n",
|
|
ISCONSTANT, get_resource_name(it), type, it->n.s_name,
|
|
get_resource_name(it), it->size );
|
|
else
|
|
fprintf(code,"%sstruct resource %s = {%d,%d,\"@%d\",%s__bytes,%d};\n",
|
|
ISCONSTANT, get_resource_name(it), it->n.i_name, type,
|
|
it->n.i_name, get_resource_name(it), it->size );
|
|
}
|
|
|
|
/* Print the resource table (NULL terminated) */
|
|
|
|
fprintf(code,"\n%sstruct resource * %sTable[] = {\n", ISCONSTANT, prefix);
|
|
for (it=top;it;it=it->next)
|
|
fprintf( code, " &%s,\n", get_resource_name(it) );
|
|
fprintf( code, " 0\n};\n" );
|
|
}
|
|
|
|
gen_res* make_font(gen_res* res)
|
|
{
|
|
fprintf(stderr,"Fonts not supported\n");
|
|
return NULL;
|
|
}
|
|
|
|
gen_res* make_raw(gen_res* res)
|
|
{
|
|
fprintf(stderr,"RCData not supported\n");
|
|
return NULL;
|
|
}
|
|
|
|
gen_res* int_to_raw(int i,gen_res* res)
|
|
{
|
|
fprintf(stderr,"IntToRaw not supported\n");
|
|
return NULL;
|
|
}
|
|
|
|
/* translate "Hello,\\tworld!\\10" to "Hello,\tworld!\n" */
|
|
char *parse_c_string(char *in)
|
|
{
|
|
char *out=xmalloc(strlen(in)-1);
|
|
char *it;
|
|
char tmp[5],*tend;
|
|
for(it=out,in++;*in;in++)
|
|
if(*in=='\\')
|
|
switch(*++in)
|
|
{case 't':*it++='\t';break;
|
|
case 'r':*it++='\r';break;
|
|
case 'n':*it++='\n';break;
|
|
case 'a':*it++='\a';break;
|
|
case '0':
|
|
memset(tmp,0,5);/*make sure it doesn't use more than 4 chars*/
|
|
memcpy(tmp,in,4);
|
|
*it++=strtoul(tmp,&tend,0);
|
|
in+=tend-tmp-1;
|
|
break;
|
|
case '1':case '2':case '3':case '4':case '5':
|
|
case '6':case '7':case '8':case '9':
|
|
memset(tmp,0,5);
|
|
memcpy(tmp,in,3);
|
|
*it++=strtoul(tmp,&tend,10);
|
|
in+=tend-tmp-1;
|
|
break;
|
|
case 'x':
|
|
memset(tmp,0,5);
|
|
memcpy(tmp,++in,2);
|
|
*it++=strtoul(tmp,&tend,16);
|
|
in+=tend-tmp-1;
|
|
break;
|
|
default:*it++=*in;
|
|
}
|
|
else
|
|
*it++=*in;
|
|
*(it-1)='\0';
|
|
return out;
|
|
}
|