/* * Copyright 2003 Vincent Béron * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #include #include "dumpres.h" #include "wrc.h" #define MASTER_LANGUAGE LANG_ENGLISH #define NB_LANG 0x94 enum lang_type_e { lang_type_master = 0, lang_type_neutral, lang_type_normal }; static int present_resources[res_usr+1]; static char *res_names[res_usr+1]; static int nb_resources[res_usr+1][lang_type_normal+1]; static resource_t **list_resources[res_usr+1][lang_type_normal+1]; static int get_language_id(resource_t *resource) { switch(resource->type) { case res_acc: return resource->res.acc->lvc.language->id; case res_bmp: return resource->res.bmp->data->lvc.language->id; case res_cur: return resource->res.cur->lvc.language->id; case res_curg: return resource->res.curg->lvc.language->id; case res_dlg: return resource->res.dlg->lvc.language->id; case res_dlgex: return resource->res.dlgex->lvc.language->id; case res_fnt: return resource->res.fnt->data->lvc.language->id; case res_fntdir: return resource->res.fnd->data->lvc.language->id; case res_ico: return resource->res.ico->lvc.language->id; case res_icog: return resource->res.icog->lvc.language->id; case res_men: return resource->res.men->lvc.language->id; case res_menex: return resource->res.menex->lvc.language->id; case res_rdt: return resource->res.rdt->data->lvc.language->id; case res_stt: return resource->res.stt->lvc.language->id; case res_usr: return resource->res.usr->data->lvc.language->id; case res_msg: return resource->res.msg->data->lvc.language->id; case res_ver: return resource->res.ver->lvc.language->id; case res_dlginit: return resource->res.dlgi->data->lvc.language->id; case res_toolbar: return resource->res.tbt->lvc.language->id; case res_anicur: case res_aniico: return resource->res.ani->data->lvc.language->id; default: /* Not supposed to reach here */ fprintf(stderr, "Not supposed to reach here (get_language_id())\n"); abort(); return -1; } } static void add_resource(resource_t *resource) { enum lang_type_e lang_type; enum res_e res_type = resource->type; int lid = get_language_id(resource); if(lid == MASTER_LANGUAGE) { lang_type = lang_type_master; } else if(lid == LANG_NEUTRAL) { lang_type = lang_type_neutral; } else { lang_type = lang_type_normal; } nb_resources[res_type][lang_type]++; list_resources[res_type][lang_type] = realloc(list_resources[res_type][lang_type], nb_resources[res_type][lang_type]*sizeof(resource_t *)); list_resources[res_type][lang_type][nb_resources[res_type][lang_type]-1] = resource; } #if 0 #define PRETTYPRINTLANG(langid) \ if(LANG_##langid == lid) { \ return #langid; \ } static const char *get_language_name(int lid) { PRETTYPRINTLANG(NEUTRAL) PRETTYPRINTLANG(AFRIKAANS) PRETTYPRINTLANG(ALBANIAN) PRETTYPRINTLANG(ARABIC) PRETTYPRINTLANG(ARMENIAN) PRETTYPRINTLANG(ASSAMESE) PRETTYPRINTLANG(AZERI) PRETTYPRINTLANG(BASQUE) PRETTYPRINTLANG(BELARUSIAN) PRETTYPRINTLANG(BENGALI) PRETTYPRINTLANG(BULGARIAN) PRETTYPRINTLANG(CATALAN) PRETTYPRINTLANG(CHINESE) PRETTYPRINTLANG(CROATIAN) PRETTYPRINTLANG(CZECH) PRETTYPRINTLANG(DANISH) PRETTYPRINTLANG(DIVEHI) PRETTYPRINTLANG(DUTCH) PRETTYPRINTLANG(ENGLISH) PRETTYPRINTLANG(ESTONIAN) PRETTYPRINTLANG(FAEROESE) PRETTYPRINTLANG(FARSI) PRETTYPRINTLANG(FINNISH) PRETTYPRINTLANG(FRENCH) PRETTYPRINTLANG(GALICIAN) PRETTYPRINTLANG(GEORGIAN) PRETTYPRINTLANG(GERMAN) PRETTYPRINTLANG(GREEK) PRETTYPRINTLANG(GUJARATI) PRETTYPRINTLANG(HEBREW) PRETTYPRINTLANG(HINDI) PRETTYPRINTLANG(HUNGARIAN) PRETTYPRINTLANG(ICELANDIC) PRETTYPRINTLANG(INDONESIAN) PRETTYPRINTLANG(ITALIAN) PRETTYPRINTLANG(JAPANESE) PRETTYPRINTLANG(KANNADA) PRETTYPRINTLANG(KASHMIRI) PRETTYPRINTLANG(KAZAK) PRETTYPRINTLANG(KONKANI) PRETTYPRINTLANG(KOREAN) PRETTYPRINTLANG(KYRGYZ) PRETTYPRINTLANG(LATVIAN) PRETTYPRINTLANG(LITHUANIAN) PRETTYPRINTLANG(MACEDONIAN) PRETTYPRINTLANG(MALAY) PRETTYPRINTLANG(MALAYALAM) PRETTYPRINTLANG(MANIPURI) PRETTYPRINTLANG(MARATHI) PRETTYPRINTLANG(MONGOLIAN) PRETTYPRINTLANG(NEPALI) PRETTYPRINTLANG(NORWEGIAN) PRETTYPRINTLANG(ORIYA) PRETTYPRINTLANG(POLISH) PRETTYPRINTLANG(PORTUGUESE) PRETTYPRINTLANG(PUNJABI) PRETTYPRINTLANG(ROMANIAN) PRETTYPRINTLANG(RUSSIAN) PRETTYPRINTLANG(SANSKRIT) PRETTYPRINTLANG(SERBIAN) PRETTYPRINTLANG(SINDHI) PRETTYPRINTLANG(SLOVAK) PRETTYPRINTLANG(SLOVENIAN) PRETTYPRINTLANG(SPANISH) PRETTYPRINTLANG(SWAHILI) PRETTYPRINTLANG(SWEDISH) PRETTYPRINTLANG(SYRIAC) PRETTYPRINTLANG(TAMIL) PRETTYPRINTLANG(TATAR) PRETTYPRINTLANG(TELUGU) PRETTYPRINTLANG(THAI) PRETTYPRINTLANG(TURKISH) PRETTYPRINTLANG(UKRAINIAN) PRETTYPRINTLANG(URDU) PRETTYPRINTLANG(UZBEK) PRETTYPRINTLANG(VIETNAMESE) PRETTYPRINTLANG(GAELIC) PRETTYPRINTLANG(MALTESE) PRETTYPRINTLANG(MAORI) PRETTYPRINTLANG(RHAETO_ROMANCE) PRETTYPRINTLANG(SAAMI) PRETTYPRINTLANG(SORBIAN) PRETTYPRINTLANG(SUTU) PRETTYPRINTLANG(TSONGA) PRETTYPRINTLANG(TSWANA) PRETTYPRINTLANG(VENDA) PRETTYPRINTLANG(XHOSA) PRETTYPRINTLANG(ZULU) PRETTYPRINTLANG(ESPERANTO) PRETTYPRINTLANG(WALON) PRETTYPRINTLANG(CORNISH) PRETTYPRINTLANG(WELSH) PRETTYPRINTLANG(BRETON) return "Unknown language"; } #endif static int compare_accelerator(accelerator_t *accelerator1, accelerator_t *accelerator2) { int different = 0; event_t *ev1 = NULL, *ev2 = NULL; if(!different && ((accelerator1->memopt != accelerator2->memopt) || (accelerator1->lvc.version != accelerator2->lvc.version) || (accelerator1->lvc.characts != accelerator2->lvc.characts))) different = 1; ev1 = accelerator1->events; ev2 = accelerator2->events; while(!different && ev1 && ev2) { if(!different && ((ev1->id != ev2->id) || (ev1->flags != ev2->flags))) different = 1; ev1 = ev1->next; ev2 = ev2->next; } if(!different && ((ev1 && !ev2) || (!ev1 && ev2))) different = 1; return different; } static int compare_bitmap(bitmap_t *bitmap1, bitmap_t *bitmap2) { int different = 0; if(!different && ((bitmap1->memopt != bitmap2->memopt) || (bitmap1->data->lvc.version != bitmap2->data->lvc.version) || (bitmap1->data->lvc.characts != bitmap2->data->lvc.characts))) different = 1; return different; } static int compare_cursor(cursor_t *cursor1, cursor_t *cursor2) { int different = 0; if(!different && ((cursor1->id != cursor2->id) || (cursor1->width != cursor2->width) || (cursor1->height != cursor2->height) || (cursor1->xhot != cursor2->xhot) || (cursor1->yhot != cursor2->yhot))) different = 1; if(!different && ((cursor1->lvc.version != cursor2->lvc.version) || (cursor1->lvc.characts != cursor2->lvc.characts))) different = 1; return different; } static int compare_cursor_group(cursor_group_t *cursor_group1, cursor_group_t *cursor_group2) { int different = 0; cursor_t *cursor1 = NULL, *cursor2 = NULL; if(!different && ((cursor_group1->memopt != cursor_group2->memopt) || (cursor_group1->lvc.version != cursor_group2->lvc.version) || (cursor_group1->lvc.characts != cursor_group2->lvc.characts))) different = 1; if(!different && (cursor_group1->ncursor != cursor_group2->ncursor)) different = 1; if(!different) { cursor1 = cursor_group1->cursorlist; cursor2 = cursor_group2->cursorlist; while(!different && cursor1 && cursor2) { different = compare_cursor(cursor1, cursor2); cursor1 = cursor1->next; cursor2 = cursor2->next; } if(!different && ((cursor1 && !cursor2) || (!cursor1 && cursor2))) different = 1; } return different; } static int compare_control(control_t *control1, control_t *control2) { int different = 0; char *nameid = NULL; if(!different && ((control1 && !control2) || (!control1 && control2))) different = 1; if(different || !control1 || !control2) return different; nameid = strdup(get_nameid_str(control1->ctlclass)); if(!different && strcmp(nameid, get_nameid_str(control2->ctlclass))) different = 1; free(nameid); if(!different && (control1->id != control2->id)) different = 1; if(!different && control1->gotstyle && control2->gotstyle) { if((!control1->style || !control2->style) || (control1->style->and_mask || control2->style->and_mask) || (control1->style->or_mask != control2->style->or_mask)) different = 1; } else if(!different && ((control1->gotstyle && !control2->gotstyle) || (!control1->gotstyle && control2->gotstyle))) different = 1; if(!different && control1->gotexstyle && control2->gotexstyle) { if((!control1->exstyle || !control2->exstyle) || (control1->exstyle->and_mask || control2->exstyle->and_mask) || (control1->exstyle->or_mask != control2->exstyle->or_mask)) different = 1; } else if(!different && ((control1->gotexstyle && !control2->gotexstyle) || (!control1->gotexstyle && control2->gotexstyle))) different = 1; if(!different && control1->gothelpid && control2->gothelpid) { if(control1->helpid != control2->helpid) different = 1; } else if(!different && ((control1->gothelpid && !control2->gothelpid) || (!control1->gothelpid && control2->gothelpid))) different = 1; return different; } static int compare_dialog(dialog_t *dialog1, dialog_t *dialog2) { int different = 0; char *nameid = NULL; if(!different && ((dialog1->memopt != dialog2->memopt) || (dialog1->lvc.version != dialog2->lvc.version) || (dialog1->lvc.characts != dialog2->lvc.characts))) different = 1; if(!different && dialog1->gotstyle && dialog2->gotstyle) { if((!dialog1->style || !dialog2->style) || (dialog1->style->and_mask || dialog2->style->and_mask) || (dialog1->style->or_mask != dialog2->style->or_mask)) different = 1; } else if(!different && ((dialog1->gotstyle && !dialog2->gotstyle) || (!dialog1->gotstyle && dialog2->gotstyle))) different = 1; if(!different && dialog1->gotexstyle && dialog2->gotexstyle) { if((!dialog1->exstyle || !dialog2->exstyle) || (dialog1->exstyle->and_mask || dialog2->exstyle->and_mask) || (dialog1->exstyle->or_mask != dialog2->exstyle->or_mask)) different = 1; } else if(!different && ((dialog1->gotexstyle && !dialog2->gotexstyle) || (!dialog1->gotexstyle && dialog2->gotexstyle))) different = 1; nameid = strdup(get_nameid_str(dialog1->menu)); if(!different && strcmp(nameid, get_nameid_str(dialog2->menu))) different = 1; free(nameid); nameid = strdup(get_nameid_str(dialog1->dlgclass)); if(!different && strcmp(nameid, get_nameid_str(dialog2->dlgclass))) different = 1; free(nameid); if(!different) different = compare_control(dialog1->controls, dialog2->controls); return different; } static int compare_dialogex(dialogex_t *dialogex1, dialogex_t *dialogex2) { int different = 0; char *nameid = NULL; if(!different && ((dialogex1->memopt != dialogex2->memopt) || (dialogex1->lvc.version != dialogex2->lvc.version) || (dialogex1->lvc.characts != dialogex2->lvc.characts))) different = 1; if(!different && dialogex1->gotstyle && dialogex2->gotstyle) { if((!dialogex1->style || !dialogex2->style) || (dialogex1->style->and_mask || dialogex2->style->and_mask) || (dialogex1->style->or_mask != dialogex2->style->or_mask)) different = 1; } else if(!different && ((dialogex1->gotstyle && !dialogex2->gotstyle) || (!dialogex1->gotstyle && dialogex2->gotstyle))) different = 1; if(!different && dialogex1->gotexstyle && dialogex2->gotexstyle) { if((!dialogex1->exstyle || !dialogex2->exstyle) || (dialogex1->exstyle->and_mask || dialogex2->exstyle->and_mask) || (dialogex1->exstyle->or_mask != dialogex2->exstyle->or_mask)) different = 1; } else if(!different && ((dialogex1->gotexstyle && !dialogex2->gotexstyle) || (!dialogex1->gotexstyle && dialogex2->gotexstyle))) different = 1; if(!different && dialogex1->gothelpid && dialogex2->gothelpid) { if(dialogex1->helpid != dialogex2->helpid) different = 1; } else if(!different && ((dialogex1->gothelpid && !dialogex2->gothelpid) || (!dialogex1->gothelpid && dialogex2->gothelpid))) different = 1; nameid = strdup(get_nameid_str(dialogex1->menu)); if(!different && strcmp(nameid, get_nameid_str(dialogex2->menu))) different = 1; free(nameid); nameid = strdup(get_nameid_str(dialogex1->dlgclass)); if(!different && strcmp(nameid, get_nameid_str(dialogex2->dlgclass))) different = 1; free(nameid); if(!different) different = compare_control(dialogex1->controls, dialogex2->controls); return different; } static int compare_font(font_t *font1, font_t *font2) { int different = 0; if(!different && ((font1->memopt != font2->memopt) || (font1->data->lvc.version != font2->data->lvc.version) || (font1->data->lvc.characts != font2->data->lvc.characts))) different = 1; return different; } static int compare_fontdir(fontdir_t *fontdir1, fontdir_t *fontdir2) { int different = 0; if(!different && ((fontdir1->memopt != fontdir2->memopt) || (fontdir1->data->lvc.version != fontdir2->data->lvc.version) || (fontdir1->data->lvc.characts != fontdir2->data->lvc.characts))) different = 1; return different; } static int compare_icon(icon_t *icon1, icon_t *icon2) { int different = 0; if(!different && ((icon1->id != icon2->id) || (icon1->width != icon2->width) || (icon1->height != icon2->height))) different = 1; if(!different && ((icon1->lvc.version != icon2->lvc.version) || (icon1->lvc.characts != icon2->lvc.characts))) different = 1; return different; } static int compare_icon_group(icon_group_t *icon_group1, icon_group_t *icon_group2) { int different = 0; icon_t *icon1 = NULL, *icon2 = NULL; if(!different && ((icon_group1->memopt != icon_group2->memopt) || (icon_group1->lvc.version != icon_group2->lvc.version) || (icon_group1->lvc.characts != icon_group2->lvc.characts))) different = 1; if(!different && (icon_group1->nicon != icon_group2->nicon)) different = 1; if(!different) { icon1 = icon_group1->iconlist; icon2 = icon_group2->iconlist; while(!different && icon1 && icon2) { different = compare_icon(icon1, icon2); icon1 = icon1->next; icon2 = icon2->next; } if(!different && ((icon1 && !icon2) || (!icon1 && icon2))) different = 1; } return different; } static int compare_menu_item(menu_item_t *menu_item1, menu_item_t *menu_item2) { int different = 0; while(!different && menu_item1 && menu_item2) { if(menu_item1->popup && menu_item2->popup) different = compare_menu_item(menu_item1->popup, menu_item2->popup); else if(!menu_item1->popup && !menu_item2->popup) { if(menu_item1->name && menu_item2->name) { if((menu_item1->id != menu_item2->id) || (menu_item1->state != menu_item2->state)) different = 1; } else if((menu_item1->name && !menu_item2->name) || (!menu_item1->name && menu_item2->name)) different = 1; } else different = 1; menu_item1 = menu_item1->next; menu_item2 = menu_item2->next; } if(!different && ((menu_item1 && !menu_item2) || (!menu_item1 && menu_item2))) different = 1; return different; } static int compare_menu(menu_t *menu1, menu_t *menu2) { int different = 0; if(!different && ((menu1->memopt != menu2->memopt) || (menu1->lvc.version != menu2->lvc.version) || (menu1->lvc.characts != menu2->lvc.characts))) different = 1; if(!different) different = compare_menu_item(menu1->items, menu2->items); return different; } static int compare_menuex_item(menuex_item_t *menuex_item1, menuex_item_t *menuex_item2) { int different = 0; while(!different && menuex_item1 && menuex_item2) { if(menuex_item1->popup && menuex_item2->popup) { if(!different && menuex_item1->gotid && menuex_item2->gotid) { if(menuex_item1->id != menuex_item2->id) different = 1; } else if(!different && ((menuex_item1->gotid && !menuex_item2->gotid) || (!menuex_item2->gotid && menuex_item2->gotid))) different = 1; if(!different && menuex_item1->gottype && menuex_item2->gottype) { if(menuex_item1->type != menuex_item2->type) different = 1; } else if(!different && ((menuex_item1->gottype && !menuex_item2->gottype) || (!menuex_item2->gottype && menuex_item2->gottype))) different = 1; if(!different && menuex_item1->gotstate && menuex_item2->gotstate) { if(menuex_item1->state != menuex_item2->state) different = 1; } else if(!different && ((menuex_item1->gotstate && !menuex_item2->gotstate) || (!menuex_item2->gotstate && menuex_item2->gotstate))) different = 1; if(!different && menuex_item1->gothelpid && menuex_item2->gothelpid) { if(menuex_item1->helpid != menuex_item2->helpid) different = 1; } else if(!different && ((menuex_item1->gothelpid && !menuex_item2->gothelpid) || (!menuex_item2->gothelpid && menuex_item2->gothelpid))) different = 1; if(!different) different = compare_menuex_item(menuex_item1->popup, menuex_item2->popup); } else if(!menuex_item1->popup && !menuex_item2->popup) { if(menuex_item1->name && menuex_item2->name) { if(!different && menuex_item1->gotid && menuex_item2->gotid) { if(menuex_item1->id != menuex_item2->id) different = 1; } else if(!different && ((menuex_item1->gotid && !menuex_item2->gotid) || (!menuex_item2->gotid && menuex_item2->gotid))) different = 1; if(!different && menuex_item1->gottype && menuex_item2->gottype) { if(menuex_item1->type != menuex_item2->type) different = 1; } else if(!different && ((menuex_item1->gottype && !menuex_item2->gottype) || (!menuex_item2->gottype && menuex_item2->gottype))) different = 1; if(!different && menuex_item1->gotstate && menuex_item2->gotstate) { if(menuex_item1->state != menuex_item2->state) different = 1; } else if(!different && ((menuex_item1->gotstate && !menuex_item2->gotstate) || (!menuex_item2->gotstate && menuex_item2->gotstate))) different = 1; if(!different && menuex_item1->gothelpid && menuex_item2->gothelpid) { if(menuex_item1->helpid != menuex_item2->helpid) different = 1; } else if(!different && ((menuex_item1->gothelpid && !menuex_item2->gothelpid) || (!menuex_item2->gothelpid && menuex_item2->gothelpid))) different = 1; } else if((menuex_item1->name && !menuex_item2->name) || (!menuex_item1->name && menuex_item2->name)) different = 1; } else different = 1; menuex_item1 = menuex_item1->next; menuex_item2 = menuex_item2->next; } if(!different && ((menuex_item1 && !menuex_item2) || (!menuex_item1 && menuex_item2))) different = 1; return different; } static int compare_menuex(menuex_t *menuex1, menuex_t *menuex2) { int different = 0; if(!different && ((menuex1->memopt != menuex2->memopt) || (menuex1->lvc.version != menuex2->lvc.version) || (menuex1->lvc.characts != menuex2->lvc.characts))) different = 1; if(!different) different = compare_menuex_item(menuex1->items, menuex2->items); return different; } static int compare_rcdata(rcdata_t *rcdata1, rcdata_t *rcdata2) { int different = 0; if(!different && ((rcdata1->memopt != rcdata2->memopt) || (rcdata1->data->lvc.version != rcdata2->data->lvc.version) || (rcdata1->data->lvc.characts != rcdata2->data->lvc.characts))) different = 1; return different; } static int compare_stringtable(stringtable_t *stringtable1, stringtable_t *stringtable2) { int different = 0; int i; while(!different && stringtable1 && stringtable2) { if((stringtable1->memopt != stringtable2->memopt) || (stringtable1->lvc.version != stringtable2->lvc.version) || (stringtable1->lvc.characts != stringtable2->lvc.characts)) different = 1; if(!different) { if((stringtable1->nentries != stringtable2->nentries) || (stringtable1->idbase != stringtable2->idbase)) different = 1; else for(i = 0 ; i < stringtable1->nentries; i++) if((stringtable1->entries[i].id != stringtable2->entries[i].id) || (stringtable1->entries[i].memopt != stringtable2->entries[i].memopt) || (stringtable1->entries[i].str && !stringtable2->entries[i].str) || (!stringtable1->entries[i].str && stringtable2->entries[i].str)) { different = 1; break; } } stringtable1 = stringtable1->next; stringtable2 = stringtable2->next; } return different; } static int compare_user(user_t *user1, user_t *user2) { int different = 0; char *nameid = NULL; if(!different && ((user1->memopt != user2->memopt) || (user1->data->lvc.version != user2->data->lvc.version) || (user1->data->lvc.characts != user2->data->lvc.characts))) different = 1; nameid = strdup(get_nameid_str(user1->type)); if(!different && strcmp(nameid, get_nameid_str(user2->type))) different = 1; free(nameid); return different; } static int compare_messagetable(messagetable_t *messagetable1, messagetable_t *messagetable2) { int different = 0; if(!different && ((messagetable1->memopt != messagetable2->memopt) || (messagetable1->data->lvc.version != messagetable2->data->lvc.version) || (messagetable1->data->lvc.characts != messagetable2->data->lvc.characts))) different = 1; return different; } static int compare_string(string_t *string1, string_t *string2) { int different = 0; if(!different && ((string1->size != string2->size) || (string1->type != string2->type))) different = 1; if(!different) { if(string1->type == str_char) different = memcmp(string1->str.cstr, string2->str.cstr, string1->size); else if(string1->type == str_unicode) different = memcmp(string1->str.wstr, string2->str.wstr, string1->size*sizeof(WCHAR)); else different = 1; } return different; } static int compare_ver_block(ver_block_t *ver_block1, ver_block_t *ver_block2); static int compare_ver_value(ver_value_t *ver_value1, ver_value_t *ver_value2) { int different = 0; int i = 0; if(!different && (ver_value1->type == ver_value2->type)) { switch(ver_value1->type) { case val_str: if(!different && ver_value1->key && ver_value2->key) different = compare_string(ver_value1->key, ver_value2->key); else if(!different && ((ver_value1->key && !ver_value2->key) || (!ver_value1->key && ver_value2->key))) different = 1; break; case val_words: if(!different && ver_value1->key && ver_value2->key) different = compare_string(ver_value1->key, ver_value2->key); else if(!different && ((ver_value1->key && !ver_value2->key) || (!ver_value1->key && ver_value2->key))) different = 1; if(!different && ver_value1->value.words && ver_value2->value.words) { if(!different && (ver_value1->value.words->nwords != ver_value2->value.words->nwords)) different = 1; if(!different) for(i = 0; i < ver_value1->value.words->nwords; i++) { if(ver_value1->value.words->words[i] != ver_value2->value.words->words[i]) { different = 1; break; } } } else if(!different && ((ver_value1->value.words && !ver_value2->value.words) || (!ver_value1->value.words && ver_value2->value.words))) different = 1; break; case val_block: if(!different && ver_value1->value.block && ver_value2->value.block) different = compare_ver_block(ver_value1->value.block, ver_value2->value.block); else if(!different && ((ver_value1->value.block && !ver_value2->value.block) || (!ver_value1->value.block && ver_value2->value.block))) different = 1; break; default: different = 1; } } else different = 1; return different; } static int compare_ver_block(ver_block_t *ver_block1, ver_block_t *ver_block2) { int different = 0; ver_value_t *ver_value1 = NULL, *ver_value2 = NULL; if(!different) { ver_value1 = ver_block1->values; ver_value2 = ver_block2->values; while(!different && ver_value1 && ver_value2) { different = compare_ver_value(ver_value1, ver_value2); ver_value1 = ver_value1->next; ver_value2 = ver_value2->next; } if(!different && ((ver_value1 && !ver_value2) || (!ver_value1 && ver_value2))) different = 1; } return different; } static int compare_versioninfo(versioninfo_t *versioninfo1, versioninfo_t *versioninfo2) { int different = 0; ver_block_t *ver_block1 = NULL, *ver_block2 = NULL; if(!different && ((versioninfo1->memopt != versioninfo2->memopt) || (versioninfo1->lvc.version != versioninfo2->lvc.version) || (versioninfo1->lvc.characts != versioninfo2->lvc.characts))) different = 1; if(!different && versioninfo1->gotit.fv && versioninfo2->gotit.fv) { if((versioninfo1->filever_maj1 != versioninfo2->filever_maj1) || (versioninfo1->filever_maj2 != versioninfo2->filever_maj2) || (versioninfo1->filever_min1 != versioninfo2->filever_min1) || (versioninfo1->filever_min2 != versioninfo2->filever_min2)) different = 1; } else if(!different && ((versioninfo1->gotit.fv && !versioninfo2->gotit.fv) || (!versioninfo1->gotit.fv && versioninfo2->gotit.fv))) different = 1; if(!different && versioninfo1->gotit.pv && versioninfo2->gotit.pv) { if((versioninfo1->prodver_maj1 != versioninfo2->prodver_maj1) || (versioninfo1->prodver_maj2 != versioninfo2->prodver_maj2) || (versioninfo1->prodver_min1 != versioninfo2->prodver_min1) || (versioninfo1->prodver_min2 != versioninfo2->prodver_min2)) different = 1; } else if(!different && ((versioninfo1->gotit.pv && !versioninfo2->gotit.pv) || (!versioninfo1->gotit.pv && versioninfo2->gotit.pv))) different = 1; if(!different && versioninfo1->gotit.fo && versioninfo2->gotit.fo) { if(versioninfo1->fileos != versioninfo2->fileos) different = 1; } else if(!different && ((versioninfo1->gotit.fo && !versioninfo2->gotit.fo) || (!versioninfo1->gotit.fo && versioninfo2->gotit.fo))) different = 1; if(!different && versioninfo1->gotit.ff && versioninfo2->gotit.ff) { if(versioninfo1->fileflags != versioninfo2->fileflags) different = 1; } else if(!different && ((versioninfo1->gotit.ff && !versioninfo2->gotit.ff) || (!versioninfo1->gotit.ff && versioninfo2->gotit.ff))) different = 1; if(!different && versioninfo1->gotit.ffm && versioninfo2->gotit.ffm) { if(versioninfo1->fileflagsmask != versioninfo2->fileflagsmask) different = 1; } else if(!different && ((versioninfo1->gotit.ffm && !versioninfo2->gotit.ffm) || (!versioninfo1->gotit.ffm && versioninfo2->gotit.ffm))) different = 1; if(!different && versioninfo1->gotit.ft && versioninfo2->gotit.ft) { if(versioninfo1->filetype != versioninfo2->filetype) different = 1; } else if(!different && ((versioninfo1->gotit.ft && !versioninfo2->gotit.ft) || (!versioninfo1->gotit.ft && versioninfo2->gotit.ft))) different = 1; if(!different && versioninfo1->gotit.fst && versioninfo2->gotit.fst) { if(versioninfo1->filesubtype != versioninfo2->filesubtype) different = 1; } else if(!different && ((versioninfo1->gotit.fst && !versioninfo2->gotit.fst) || (!versioninfo1->gotit.fst && versioninfo2->gotit.fst))) different = 1; if(!different) { ver_block1 = versioninfo1->blocks; ver_block2 = versioninfo2->blocks; while(!different && ver_block1 && ver_block2) { different = compare_ver_block(ver_block1, ver_block2); ver_block1 = ver_block1->next; ver_block2 = ver_block2->next; } if(!different && ((ver_block1 && !ver_block2) || (ver_block1 && !ver_block2))) different = 1; } return different; } static int compare_dlginit(dlginit_t *dlginit1, dlginit_t *dlginit2) { int different = 0; if(!different && ((dlginit1->memopt != dlginit2->memopt) || (dlginit1->data->lvc.version != dlginit2->data->lvc.version) || (dlginit1->data->lvc.characts != dlginit2->data->lvc.characts))) different = 1; return different; } static int compare_toolbar_item(toolbar_item_t *toolbar_item1, toolbar_item_t *toolbar_item2) { int different = 0; while(!different && toolbar_item1 && toolbar_item2) { if((toolbar_item1->id && !toolbar_item2->id) || (!toolbar_item1->id && toolbar_item2->id)) different = 1; toolbar_item1 = toolbar_item1->next; toolbar_item2 = toolbar_item2->next; } if(!different && ((toolbar_item1 && !toolbar_item2) || (!toolbar_item1 && toolbar_item2))) different = 1; return different; } static int compare_toolbar(toolbar_t *toolbar1, toolbar_t *toolbar2) { int different = 0; if(!different && ((toolbar1->memopt != toolbar2->memopt) || (toolbar1->lvc.version != toolbar2->lvc.version) || (toolbar1->lvc.characts != toolbar2->lvc.characts))) different = 1; if(!different) different = compare_toolbar_item(toolbar1->items, toolbar2->items); return different; } static int compare_ani_curico(ani_curico_t *ani_curico1, ani_curico_t *ani_curico2) { int different = 0; if(!different && ((ani_curico1->memopt != ani_curico2->memopt) || (ani_curico1->data->lvc.version != ani_curico2->data->lvc.version) || (ani_curico1->data->lvc.characts != ani_curico2->data->lvc.characts))) different = 1; return different; } static int compare(resource_t *resource1, resource_t *resource2) { switch(resource1->type) { case res_acc: return compare_accelerator(resource1->res.acc, resource2->res.acc); case res_bmp: return compare_bitmap(resource1->res.bmp, resource2->res.bmp); case res_cur: return compare_cursor(resource1->res.cur, resource2->res.cur); case res_curg: return compare_cursor_group(resource1->res.curg, resource2->res.curg); case res_dlg: return compare_dialog(resource1->res.dlg, resource2->res.dlg); case res_dlgex: return compare_dialogex(resource1->res.dlgex, resource2->res.dlgex); case res_fnt: return compare_font(resource1->res.fnt, resource2->res.fnt); case res_fntdir: return compare_fontdir(resource1->res.fnd, resource2->res.fnd); case res_ico: return compare_icon(resource1->res.ico, resource2->res.ico); case res_icog: return compare_icon_group(resource1->res.icog, resource2->res.icog); case res_men: return compare_menu(resource1->res.men, resource2->res.men); case res_menex: return compare_menuex(resource1->res.menex, resource2->res.menex); case res_rdt: return compare_rcdata(resource1->res.rdt, resource2->res.rdt); case res_stt: return compare_stringtable(resource1->res.stt, resource2->res.stt); case res_usr: return compare_user(resource1->res.usr, resource2->res.usr); case res_msg: return compare_messagetable(resource1->res.msg, resource2->res.msg); case res_ver: return compare_versioninfo(resource1->res.ver, resource2->res.ver); case res_dlginit: return compare_dlginit(resource1->res.dlgi, resource2->res.dlgi); case res_toolbar: return compare_toolbar(resource1->res.tbt, resource2->res.tbt); case res_anicur: case res_aniico: return compare_ani_curico(resource1->res.ani, resource2->res.ani); default: /* Not supposed to reach here */ fprintf(stderr, "Not supposed to reach here (compare())\n"); abort(); return -1; } } void verify_translations(resource_t *top) { enum lang_type_e lang_type; enum res_e res_type; int **presence; int i, j; char *nameid; char **problems; int nb_problems, last_problem; int complete, needs_work, partial; resource_t *next = top; for(res_type = res_0; res_type <= res_usr; res_type++) { present_resources[res_type] = 0; for(lang_type = lang_type_master; lang_type <= lang_type_normal; lang_type++) { nb_resources[res_type][lang_type] = 0; list_resources[res_type][lang_type] = NULL; } } while(next) { switch(next->type) { case res_acc: case res_bmp: case res_cur: case res_curg: case res_dlg: case res_dlgex: case res_fnt: case res_fntdir: case res_ico: case res_icog: case res_men: case res_menex: case res_rdt: case res_stt: case res_usr: case res_msg: case res_ver: case res_dlginit: case res_toolbar: case res_anicur: case res_aniico: add_resource(next); break; default: fprintf(stderr, "Report this: unknown resource type parsed %08x\n", next->type); } next = next->next; } present_resources[res_acc] = 1; res_names[res_acc] = strdup("accelerator"); present_resources[res_bmp] = 1; res_names[res_bmp] = strdup("bitmap"); present_resources[res_cur] = 1; res_names[res_cur] = strdup("cursor"); present_resources[res_curg] = 1; res_names[res_curg] = strdup("cursor_group"); present_resources[res_dlg] = 1; res_names[res_dlg] = strdup("dialog"); present_resources[res_dlgex] = 1; res_names[res_dlgex] = strdup("dialogex"); present_resources[res_fnt] = 1; res_names[res_fnt] = strdup("font"); present_resources[res_fntdir] = 1; res_names[res_fntdir] = strdup("fontdir"); present_resources[res_ico] = 1; res_names[res_ico] = strdup("icon"); present_resources[res_icog] = 1; res_names[res_icog] = strdup("icon_group"); present_resources[res_men] = 1; res_names[res_men] = strdup("menu"); present_resources[res_menex] = 1; res_names[res_menex] = strdup("menuex"); present_resources[res_rdt] = 1; res_names[res_rdt] = strdup("rcdata"); present_resources[res_stt] = 1; res_names[res_stt] = strdup("stringtable"); present_resources[res_usr] = 1; res_names[res_usr] = strdup("user"); present_resources[res_msg] = 1; res_names[res_msg] = strdup("messagetable"); present_resources[res_ver] = 1; res_names[res_ver] = strdup("versioninfo"); present_resources[res_dlginit] = 1; res_names[res_dlginit] = strdup("dlginit"); present_resources[res_toolbar] = 1; res_names[res_toolbar] = strdup("toolbar"); present_resources[res_anicur] = 1; res_names[res_anicur] = strdup("ani_cursor"); present_resources[res_aniico] = 1; res_names[res_aniico] = strdup("ani_icon"); for(res_type = res_0; res_type <= res_usr; res_type++) { if(!present_resources[res_type]) { continue; } if(nb_resources[res_type][lang_type_normal] > 0) { if(nb_resources[res_type][lang_type_master] && nb_resources[res_type][lang_type_neutral]) { fprintf(stderr, "Type %s:\n", res_names[res_type]); fprintf(stderr, "There are both a NEUTRAL and a MASTER version for %s, along with additional localized versions. The NEUTRAL versions will not be checked against other versions.\n", res_names[res_type]); } else if(nb_resources[res_type][lang_type_neutral]) { fprintf(stderr, "Type %s:\n", res_names[res_type]); fprintf(stderr, "There are no MASTER version, but there are some NEUTRAL versions for %s, so will use those instead of MASTER for comparison.\n", res_names[res_type]); list_resources[res_type][lang_type_master] = list_resources[res_type][lang_type_neutral]; nb_resources[res_type][lang_type_master] = nb_resources[res_type][lang_type_neutral]; } else if(!nb_resources[res_type][lang_type_master]) { fprintf(stderr, "Type %s:\n", res_names[res_type]); fprintf(stderr, "There are no NEUTRAL nor MASTER versions for %s, but there are some other localized versions. No comparison will be done at all.\n", res_names[res_type]); } } else { if(nb_resources[res_type][lang_type_master] && nb_resources[res_type][lang_type_neutral]) { fprintf(stderr, "Type %s:\n", res_names[res_type]); fprintf(stderr, "There are both a NEUTRAL and a MASTER versions for %s, but no other localized version. No comparison will be done at all.\n", res_names[res_type]); } else if(nb_resources[res_type][lang_type_master]) { fprintf(stderr, "Type %s:\n", res_names[res_type]); fprintf(stderr, "There are only MASTER versions for %s. No comparison will be done at all.\n", res_names[res_type]); } else if(nb_resources[res_type][lang_type_neutral]) { /* fprintf(stderr, "There are only NEUTRAL versions for %s. No comparison will be done at all.\n", res_names[res_type]); */ } else { /* fprintf(stderr, "There are no versions at all for %s. No comparison will be done at all.\n", res_names[res_type]); */ } } presence = malloc(nb_resources[res_type][lang_type_master]*sizeof(int *)); for(i = 0; i < nb_resources[res_type][lang_type_master]; i++) { presence[i] = calloc(NB_LANG, sizeof(int)); presence[i][MASTER_LANGUAGE] = -1; } for(i = 0; i < nb_resources[res_type][lang_type_normal]; i++) { for(j = 0; j < nb_resources[res_type][lang_type_master]; j++) { nameid = strdup(get_nameid_str(list_resources[res_type][lang_type_normal][i]->name)); if(!strcmp(nameid, get_nameid_str(list_resources[res_type][lang_type_master][j]->name))) { if(compare(list_resources[res_type][lang_type_normal][i], list_resources[res_type][lang_type_master][j])) { presence[j][get_language_id(list_resources[res_type][lang_type_normal][i])] = 2; /* fprintf(stderr, "Differences in type %s, ID %s, for language %s\n", res_names[res_type], nameid, get_language_name(get_language_id(list_resources[res_type][lang_type_normal][i]))); */ } else { presence[j][get_language_id(list_resources[res_type][lang_type_normal][i])] = 1; } } free(nameid); } } problems = malloc(sizeof(char *)); problems[0] = strdup(""); nb_problems = 0; last_problem = -1; for(i = 0; i < NB_LANG; i++) { complete = 1; needs_work = 0; partial = 0; for(j = 0; j < nb_resources[res_type][lang_type_master]; j++) { if(presence[j][i]) { partial = 1; if(presence[j][i] == 2) { needs_work = 1; problems = realloc(problems, (++nb_problems+1)*sizeof(char *)); problems[nb_problems] = malloc(strlen(get_nameid_str(list_resources[res_type][lang_type_master][j]->name)) + 9); sprintf(problems[nb_problems], "DIFF %s %02x", get_nameid_str(list_resources[res_type][lang_type_master][j]->name), i); if(last_problem == i) { problems[nb_problems-1] = realloc(problems[nb_problems-1], strlen(problems[nb_problems-1]) + 3); strcat(problems[nb_problems-1], " \\"); } else { last_problem = i; } } } else { complete = 0; problems = realloc(problems, (++nb_problems+1)*sizeof(char *)); problems[nb_problems] = malloc(strlen(get_nameid_str(list_resources[res_type][lang_type_master][j]->name)) + 8); sprintf(problems[nb_problems], "ABS %s %02x", get_nameid_str(list_resources[res_type][lang_type_master][j]->name), i); if(last_problem == i) { problems[nb_problems-1] = realloc(problems[nb_problems-1], strlen(problems[nb_problems-1]) + 3); strcat(problems[nb_problems-1], " \\"); } else { last_problem = i; } } } if(complete && partial && !needs_work) { /* Support is complete, no need to do anything */ /* fprintf(stderr, "Support for language %s is complete for %s.\n", get_language_name(i), res_names[res_type]); */ printf("."); } else if(complete && partial && needs_work) { /* Support is incomplete (differing resources), needs work */ /* fprintf(stderr, "Support for language %s is incomplete (differing resources) for %s.\n", get_language_name(i), res_names[res_type]); */ printf("x"); } else if(!complete && partial && !needs_work) { /* Support is incomplete (missing resources), needs work */ /* fprintf(stderr, "Support for language %s is incomplete (missing resources) for %s.\n", get_language_name(i), res_names[res_type]); */ printf("-"); } else if(!complete && partial && needs_work) { /* Support is incomplete (missing and differing resources), needs work */ /* fprintf(stderr, "Support for language %s is incomplete (missing and differing resources) for %s.\n", get_language_name(i), res_names[res_type]); */ printf("+"); } else if(!complete && !partial) { /* Support is totally absent, might be interesting to do */ /* fprintf(stderr, "Support for language %s is absent for %s.\n", get_language_name(i), res_names[res_type]); */ printf(" "); } else { /* Support is not relevant, no need to do anything */ /* fprintf(stderr, "Support for language %s is not relevant for %s.\n", get_language_name(i), res_names[res_type]); */ printf("n"); } } printf("\n"); for(i = 1; i <= nb_problems; i++) { printf("%s\n", problems[i]); free(problems[i]); } free(problems[0]); free(problems); for(i = 0; i < nb_resources[res_type][lang_type_master]; i++) free(presence[i]); free(presence); } }