Improved decompilation to support the built-in types: app_flags, app_signature,

file_types, mini_icon, large_icon.
Only app_version is desperately missing :-)


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@17823 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2006-06-13 15:03:07 +00:00
parent 128277c969
commit 0e50eab75e

View File

@ -22,6 +22,7 @@
#include <Mime.h> #include <Mime.h>
#include <Resources.h> #include <Resources.h>
#include <Roster.h>
#include <TypeConstants.h> #include <TypeConstants.h>
#include <ctype.h> #include <ctype.h>
@ -44,7 +45,9 @@ static FILE *sHeaderFile;
static int32 sTabs; static int32 sTabs;
static bool sBraceOnNextLine = false; static bool sBraceOnNextLine = false;
static void examine_data(const char *, type_code, const void *, size_t);
static void write_generic_data(const char *name, type_code type,
const void *data, size_t length);
static void static void
@ -157,6 +160,28 @@ has_prefix(const char *name)
} }
static bool
is_string(const void *data, size_t length)
{
// We consider the buffer a string if it contains only human readable
// characters. The buffer should also end with a '\0'. Although the
// compiler allows string literals to contain embedded '\0' chars as
// well, we don't allow them here (because they may cause false hits).
if (length == 0)
return false;
char *ptr = (char *)data;
for (size_t t = 0; t < length - 1; ++t) {
if (!isprint(*ptr++))
return false;
}
return (*ptr == '\0');
}
static void static void
write_rsrc(type_code type, int32 id, const char *name) write_rsrc(type_code type, int32 id, const char *name)
{ {
@ -180,6 +205,9 @@ write_rsrc(type_code type, int32 id, const char *name)
} }
// #pragma mark - generic types
static uint8 * static uint8 *
write_raw_line(uint8 *ptr, uint8 *end, size_t bytesPerLine) write_raw_line(uint8 *ptr, uint8 *end, size_t bytesPerLine)
{ {
@ -455,8 +483,8 @@ write_rgb(const char *name, const void *data)
} }
static char * static const char *
write_string_line(char *ptr, char *end, size_t charsPerLine) write_string_line(const char *ptr, const char *end, size_t charsPerLine)
{ {
uint32 count = 0; uint32 count = 0;
bool end_of_item = false; bool end_of_item = false;
@ -502,8 +530,8 @@ static void
write_string(const char *name, type_code type, write_string(const char *name, type_code type,
const void *data, size_t length) const void *data, size_t length)
{ {
char *ptr = (char *)data; const char *ptr = (const char *)data;
char *end = ptr + length; const char *end = ptr + length;
size_t charsPerLine = 64; size_t charsPerLine = 64;
// We write an "array" resource if the string has more than 64 // We write an "array" resource if the string has more than 64
@ -573,7 +601,7 @@ write_fields(BMessage &msg)
fprintf(sOutputFile, ",\n"); fprintf(sOutputFile, ",\n");
indent(); indent();
examine_data(name, type, data, length); write_generic_data(name, type, data, length);
} }
} }
@ -630,28 +658,6 @@ write_message(const char *name, BMessage &msg, type_code type)
} }
static bool
is_string(const void *data, size_t length)
{
// We consider the buffer a string if it contains only human readable
// characters. The buffer should also end with a '\0'. Although the
// compiler allows string literals to contain embedded '\0' chars as
// well, we don't allow them here (because they may cause false hits).
if (length == 0)
return false;
char *ptr = (char *)data;
for (size_t t = 0; t < length - 1; ++t) {
if (!isprint(*ptr++))
return false;
}
return (*ptr == '\0');
}
static void static void
write_other(const char *name, type_code type, write_other(const char *name, type_code type,
const void *data, size_t length) const void *data, size_t length)
@ -666,8 +672,62 @@ write_other(const char *name, type_code type,
} }
// #pragma mark - special types
static void static void
examine_data(const char *name, type_code type, write_app_signature(const void *data, size_t length)
{
fprintf(sOutputFile, "resource app_signature ");
write_string_line((const char *)data, (const char *)data + length, length * 2);
}
static void
write_app_flags(const void *data, size_t length)
{
fprintf(sOutputFile, "resource app_flags ");
uint32 flags = *(uint32 *)data;
switch (flags & B_LAUNCH_MASK) {
case B_SINGLE_LAUNCH:
fputs("B_SINGLE_LAUNCH", sOutputFile);
break;
case B_MULTIPLE_LAUNCH:
fputs("B_MULTIPLE_LAUNCH", sOutputFile);
break;
case B_EXCLUSIVE_LAUNCH:
fputs("B_EXCLUSIVE_LAUNCH", sOutputFile);
break;
}
if (flags & B_BACKGROUND_APP)
fputs(" | B_BACKGROUND_APP", sOutputFile);
if (flags & B_ARGV_ONLY)
fputs(" | B_ARGV_ONLY", sOutputFile);
}
static void
write_app_icon(uint32 which, const void *data, size_t length)
{
fprintf(sOutputFile, "resource %s_icon ", which == B_MINI_ICON ? "mini" : "large");
write_raw(NULL, B_RAW_TYPE, data, length, which);
}
static void
write_app_file_types(const void *data, size_t length)
{
fputs("resource file_types ", sOutputFile);
write_other(NULL, B_MESSAGE_TYPE, data, length);
}
// #pragma mark - file examination
static void
write_generic_data(const char *name, type_code type,
const void *data, size_t length) const void *data, size_t length)
{ {
switch (type) { switch (type) {
@ -691,18 +751,72 @@ examine_data(const char *name, type_code type,
case B_RECT_TYPE: write_rect(name, data); break; case B_RECT_TYPE: write_rect(name, data); break;
case B_RGB_COLOR_TYPE: write_rgb(name, data); break; case B_RGB_COLOR_TYPE: write_rgb(name, data); break;
case B_MIME_STRING_TYPE: write_string(name, type, data, length); break; case B_MIME_STRING_TYPE:
case B_STRING_TYPE: write_string(name, type, data, length); break; case B_STRING_TYPE:
write_string(name, type, data, length);
break;
case B_POINTER_TYPE: write_raw(name, type, data, length); break; case B_POINTER_TYPE: write_raw(name, type, data, length); break;
case 'MICN': write_raw(name, type, data, length, 16); break; case 'MICN': write_raw(name, type, data, length, 16); break;
case 'ICON': write_raw(name, type, data, length); break; case 'ICON': write_raw(name, type, data, length); break;
default: write_other(name, type, data, length); break; default:
write_other(name, type, data, length);
break;
} }
} }
static void
write_data(int32 id, const char *name, type_code type,
const void *data, size_t length)
{
// check for special types
switch (type) {
case B_MIME_STRING_TYPE:
if (!strcmp(name, "BEOS:APP_SIG")) {
write_app_signature(data, length);
return;
}
break;
case 'MICN':
if (!strcmp(name, "BEOS:M:STD_ICON")) {
write_app_icon(B_MINI_ICON, data, length);
return;
}
break;
case 'ICON':
if (!strcmp(name, "BEOS:L:STD_ICON")) {
write_app_icon(B_LARGE_ICON, data, length);
return;
}
break;
case B_MESSAGE_TYPE:
if (!strcmp(name, "BEOS:FILE_TYPES")) {
write_app_file_types(data, length);
return;
}
break;
case 'APPF':
if (length == 4) {
write_app_flags(data, length);
return;
}
break;
}
// write generic types
write_rsrc(type, id, name);
write_generic_data(NULL, type, data, length);
}
static void static void
examine_file(char *fileName) examine_file(char *fileName)
{ {
@ -733,8 +847,7 @@ examine_file(char *fileName)
data = res.LoadResource(type, id, NULL); data = res.LoadResource(type, id, NULL);
if (data != NULL) { if (data != NULL) {
fprintf(sOutputFile, "\n"); fprintf(sOutputFile, "\n");
write_rsrc(type, id, name); write_data(id, name, type, data, length);
examine_data(NULL, type, data, length);
fprintf(sOutputFile, ";\n"); fprintf(sOutputFile, ";\n");
} }