00001
00002
00003
00004
00005
00006
00007
00008 #include <stdio.h>
00009 #include <stdlib.h>
00010
00011
00012
00013 typedef unsigned char byte;
00014 typedef unsigned short u16b;
00015 typedef unsigned long u32b;
00016
00017
00018 typedef struct usecode_func usecode_func;
00019
00020 struct usecode_func
00021 {
00022
00023 u32b where;
00024
00025
00026 u16b size;
00027
00028
00029 u16b num_call;
00030
00031
00032 u16b *called;
00033
00034
00035 u16b visited;
00036 };
00037
00038
00039 usecode_func functions[4096];
00040
00041
00042
00043
00044 void read_u16b(FILE *fff, u16b *n)
00045 {
00046 u16b x;
00047 byte c;
00048
00049 c = fgetc(fff);
00050
00051 x = c;
00052
00053 c = fgetc(fff);
00054
00055 x |= c << 8;
00056
00057 *n = x;
00058 }
00059
00060
00061
00062
00063 void read_u32b(FILE *fff, u32b *n)
00064 {
00065 u32b x;
00066 byte c;
00067
00068 c = fgetc(fff);
00069
00070 x = c;
00071
00072 c = fgetc(fff);
00073
00074 x |= c << 8;
00075
00076 c = fgetc(fff);
00077
00078 x |= c << 16;
00079
00080 c = fgetc(fff);
00081
00082 x |= c << 24;
00083
00084 *n = x;
00085 }
00086
00087
00088
00089
00090 void write_u16b(FILE *fff, u16b n)
00091 {
00092 byte c;
00093
00094 c = n & 0xFF;
00095
00096 fputc(c, fff);
00097
00098 n >>= 8;
00099
00100 c = n & 0xFF;
00101
00102 fputc(c, fff);
00103 }
00104
00105
00106
00107
00108 void write_u32b(FILE *fff, u32b n)
00109 {
00110 byte c;
00111
00112 c = n & 0xFF;
00113
00114 fputc(c, fff);
00115
00116 n >>= 8;
00117
00118 c = n & 0xFF;
00119
00120 fputc(c, fff);
00121
00122 n >>= 8;
00123
00124 c = n & 0xFF;
00125
00126 fputc(c, fff);
00127
00128 n >>= 8;
00129
00130 c = n & 0xFF;
00131
00132 fputc(c, fff);
00133 }
00134
00135
00136
00137
00138 u16b get_total_size(u16b *call_tree, u16b tree_size)
00139 {
00140 usecode_func *u_ptr;
00141 u16b size = 0;
00142 int i;
00143
00144
00145 for (i = 0; i < tree_size; i++)
00146 {
00147
00148 u_ptr = &functions[call_tree[i]];
00149
00150
00151 size += u_ptr->size;
00152 }
00153
00154
00155 return size;
00156 }
00157
00158
00159
00160
00161
00162
00163 u16b *get_tree(u16b func_num, u16b *num)
00164 {
00165 usecode_func *u_ptr;
00166 u16b *sub_tree, sub_size;
00167 u16b *our_tree, our_size;
00168 int i, j;
00169
00170
00171 u_ptr = &functions[func_num];
00172
00173
00174 if (u_ptr->visited)
00175 {
00176
00177 *num = 0;
00178
00179 return NULL;
00180 }
00181
00182
00183 our_size = 1;
00184
00185
00186 our_tree = (u16b *)malloc(sizeof(u16b));
00187
00188
00189 our_tree[0] = func_num;
00190
00191
00192 u_ptr->visited = 1;
00193
00194
00195 for (i = 0; i < u_ptr->num_call; i++)
00196 {
00197
00198 sub_tree = get_tree(u_ptr->called[i], &sub_size);
00199
00200
00201 if (!sub_size) continue;
00202
00203
00204 our_size += sub_size;
00205
00206
00207 our_tree = (u16b *)realloc(our_tree, sizeof(u16b) * our_size);
00208
00209
00210 for (j = 0; j < sub_size; j++)
00211 {
00212
00213 our_tree[our_size - j - 1] = sub_tree[j];
00214 }
00215
00216
00217 free(sub_tree);
00218 }
00219
00220
00221 *num = our_size;
00222
00223 return our_tree;
00224 }
00225
00226
00227
00228
00229 void clear_visited(void)
00230 {
00231 int i;
00232
00233
00234 for (i = 0; i < 4096; i++)
00235 {
00236
00237 functions[i].visited = 0;
00238 }
00239 }
00240
00241
00242
00243
00244 int comp_func(const void *one, const void *two)
00245 {
00246 u16b first = *((u16b *)one);
00247 u16b second = *((u16b *)two);
00248
00249
00250 if (first == second) return 0;
00251
00252
00253 if (first < second) return -1;
00254
00255
00256 return 1;
00257 }
00258
00259
00260
00261
00262 void fix_tree(u16b *tree, u16b *tree_size)
00263 {
00264
00265 qsort(tree, *tree_size, sizeof(u16b), comp_func);
00266 }
00267
00268
00269
00270
00271 int main(void)
00272 {
00273 FILE *usecode, *linkdep1, *linkdep2;
00274 u16b func_num, data_size, max_func;
00275 u16b total_size, lnk2_written = 0;
00276 u16b *func_tree, tree_size;
00277 usecode_func *u_ptr;
00278 int i, j;
00279
00280
00281 usecode = fopen("usecode", "rb");
00282
00283
00284 if (!usecode)
00285 {
00286
00287 fprintf(stderr, "Could not open usecode!\n");
00288
00289
00290 exit(1);
00291 }
00292
00293
00294 linkdep1 = fopen("linkdep1", "wb");
00295 linkdep2 = fopen("linkdep2", "wb");
00296
00297
00298 read_u16b(usecode, &func_num);
00299
00300
00301 while (!feof(usecode))
00302 {
00303
00304 u_ptr = &functions[func_num];
00305
00306
00307 u_ptr->where = ftell(usecode) - 2;
00308
00309
00310 read_u16b(usecode, &u_ptr->size);
00311
00312
00313 read_u16b(usecode, &data_size);
00314
00315
00316 fseek(usecode, data_size, SEEK_CUR);
00317
00318
00319 fseek(usecode, 4, SEEK_CUR);
00320
00321
00322 read_u16b(usecode, &u_ptr->num_call);
00323
00324
00325 u_ptr->called = (u16b *)malloc(sizeof(u16b) * u_ptr->num_call);
00326
00327
00328 for (i = 0; i < u_ptr->num_call; i++)
00329 {
00330
00331 read_u16b(usecode, &u_ptr->called[i]);
00332 }
00333
00334
00335 fseek(usecode, u_ptr->where + u_ptr->size + 4, SEEK_SET);
00336
00337
00338 max_func = func_num;
00339
00340
00341 read_u16b(usecode, &func_num);
00342 }
00343
00344
00345 for (i = 0; i <= max_func; i++)
00346 {
00347
00348 u_ptr = &functions[i];
00349
00350
00351 if (!u_ptr->size)
00352 {
00353
00354 write_u16b(linkdep1, lnk2_written);
00355 write_u16b(linkdep1, 0xffff);
00356
00357
00358 continue;
00359 }
00360
00361
00362 func_tree = get_tree(i, &tree_size);
00363
00364
00365 fix_tree(func_tree, &tree_size);
00366
00367
00368 clear_visited();
00369
00370
00371 total_size = get_total_size(func_tree, tree_size);
00372
00373
00374 write_u16b(linkdep1, lnk2_written);
00375 write_u16b(linkdep1, total_size);
00376
00377
00378 for (j = 0; j < tree_size; j++)
00379 {
00380
00381 write_u32b(linkdep2, functions[func_tree[j]].where);
00382
00383
00384 lnk2_written++;
00385 }
00386
00387
00388 free(func_tree);
00389 }
00390
00391
00392 write_u16b(linkdep1, lnk2_written);
00393 write_u16b(linkdep1, 0);
00394
00395
00396 fclose(usecode);
00397 fclose(linkdep1);
00398 fclose(linkdep2);
00399
00400
00401 return 0;
00402 }