4 * Very simple program to dump the contents of an OMF (OBJ) file
6 * This assumes a littleendian, unaligned-load-capable host and a
7 * C compiler which handles basic C99.
23 static const char *record_types
[256] =
67 typedef void (*dump_func
)(uint8_t, const uint8_t *, size_t);
69 /* Ordered collection type */
71 size_t n
; /* Elements in collection (not including 0) */
72 size_t s
; /* Elements allocated (not including 0) */
73 const void **p
; /* Element pointers */
76 struct collection c_names
, c_lsegs
, c_groups
, c_extsym
;
78 static void nomem(void)
80 fprintf(stderr
, "%s: memory allocation error\n", progname
);
85 static void add_collection(struct collection
*c
, const void *p
)
88 size_t cs
= c
->s
? (c
->s
<< 1) : INIT_SIZE
;
89 const void **cp
= realloc(c
->p
, cs
*sizeof(const void *));
97 memset(cp
+ c
->n
, 0, (cs
- c
->n
)*sizeof(const void *));
103 static const void *get_collection(struct collection
*c
, size_t index
)
111 static void hexdump_data(unsigned int offset
, const uint8_t *data
,
112 size_t n
, size_t field
)
116 for (i
= 0; i
< n
; i
+= 16) {
117 printf(" %04x: ", i
+offset
);
118 for (j
= 0; j
< 16; j
++) {
119 char sep
= (j
== 7) ? '-' : ' ';
121 printf("%02x%c", data
[i
+j
], sep
);
123 printf("xx%c", sep
); /* Beyond end of... */
125 printf(" "); /* No separator */
128 for (j
= 0; j
< 16; j
++) {
130 putchar((i
+j
>= field
) ? 'x' :
131 isprint(data
[i
+j
]) ? data
[i
+j
] : '.');
137 static void dump_unknown(uint8_t type
, const uint8_t *data
, size_t n
)
140 hexdump_data(0, data
, n
, n
);
143 static void print_dostime(const uint8_t *p
)
145 uint16_t da
= (p
[3] << 8) + p
[2];
146 uint16_t ti
= (p
[1] << 8) + p
[0];
148 printf("%04u-%02u-%02u %02u:%02u:%02u",
149 (da
>> 9) + 1980, (da
>> 5) & 15, da
& 31,
150 (ti
>> 11), (ti
>> 5) & 63, (ti
<< 1) & 63);
153 static void dump_coment_depfile(uint8_t type
, const uint8_t *data
, size_t n
)
155 if (n
> 4 && data
[4] == n
-5) {
158 printf(" %.*s\n", n
-5, data
+5);
161 hexdump_data(2, data
, n
, n
);
164 static const dump_func dump_coment_class
[256] = {
165 [0xe9] = dump_coment_depfile
168 static void dump_coment(uint8_t type
, const uint8_t *data
, size_t n
)
171 static const char *coment_class
[256] = {
172 [0x00] = "Translator",
173 [0x01] = "Copyright",
174 [0x81] = "Library specifier",
175 [0x9c] = "MS-DOS version",
176 [0x9d] = "Memory model",
178 [0x9f] = "Library search",
179 [0xa0] = "OMF extensions",
180 [0xa1] = "New OMF extension",
181 [0xa2] = "Link pass separator",
191 [0xdd] = "Timestamp",
193 [0xe3] = "Type definition",
195 [0xe9] = "Dependency file",
196 [0xff] = "Command line"
200 hexdump_data(type
, data
, 2, n
);
207 printf(" [NP=%d NL=%d UD=%02X] %02X %s\n",
212 coment_class
[class] ? coment_class
[class] : "???");
214 if (dump_coment_class
[class])
215 dump_coment_class
[class](class, data
+2, n
-2);
217 hexdump_data(2, data
+2, n
-2, n
-2);
220 /* Parse an index field */
221 static uint16_t get_index(const uint8_t **pp
)
227 return ((c
& 0x7f) << 8) + *(*pp
)++;
233 static uint16_t get_16(const uint8_t **pp
)
235 uint16_t v
= *(const uint16_t *)(*pp
);
241 static uint32_t get_32(const uint8_t **pp
)
243 const uint32_t v
= *(const uint32_t *)(*pp
);
249 /* Returns a name as a C string in a newly allocated buffer */
250 char *lname(int index
)
253 const char *p
= get_collection(&c_names
, index
);
271 /* LNAMES or LLNAMES */
272 static void dump_lnames(uint8_t type
, const uint8_t *data
, size_t n
)
274 const uint8_t *p
= data
;
275 const uint8_t *end
= data
+ n
;
280 add_collection(&c_names
, NULL
);
281 printf(" # %4u 0x%04x: \"%.*s... <%zu missing bytes>\n",
282 c_names
.n
, c_names
.n
, n
-1, p
+1, l
-n
);
284 add_collection(&c_names
, p
);
285 printf(" # %4u 0x%04x: \"%.*s\"\n",
286 c_names
.n
, c_names
.n
, l
-1, p
+1);
288 hexdump_data(p
-data
, p
, l
, n
);
294 /* SEGDEF16 or SEGDEF32 */
295 static void dump_segdef(uint8_t type
, const uint8_t *data
, size_t n
)
298 const uint8_t *p
= data
;
299 const uint8_t *end
= data
+n
;
301 static const char * const alignment
[8] =
302 { "ABS", "BYTE", "WORD", "PARA", "PAGE", "DWORD", "LTL", "?ALIGN" };
303 static const char * const combine
[8] =
304 { "PRIVATE", "?COMMON", "PUBLIC", "?COMBINE", "?PUBLIC", "STACK", "COMMON", "?PUBLIC" };
313 printf(" # %s (A%u) %s (C%u) %s%s",
314 alignment
[(attr
>> 5) & 7], (attr
>> 5) & 7,
315 combine
[(attr
>> 2) & 7], (attr
>> 2) & 7,
316 (attr
& 0x02) ? "MAXSIZE " : "",
317 (attr
& 0x01) ? "USE32" : "USE16");
319 if (((attr
>> 5) & 7) == 0) {
320 /* Absolute segment */
323 printf(" AT %04x:", get_16(&p
));
324 printf("%02x", *p
++);
330 printf(" size 0x%08x", get_32(&p
));
334 printf(" size 0x%04x", get_16(&p
));
341 printf(" name '%s'", s
);
347 printf(" class '%s'", s
);
353 printf(" ovl '%s'", s
);
357 hexdump_data(0, data
, n
, n
);
360 /* FIXUPP16 or FIXUPP32 */
361 static void dump_fixupp(uint8_t type
, const uint8_t *data
, size_t n
)
364 const uint8_t *p
= data
;
365 const uint8_t *end
= data
+ n
;
366 static const char * const method_base
[4] =
367 { "SEGDEF", "GRPDEF", "EXTDEF", "frame#" };
370 const uint8_t *start
= p
;
377 bool frame
= !!(op
& 0x40);
379 printf(" THREAD %-7s%d%s method %c%d (%s)",
380 frame
? "frame" : "target", op
& 3,
381 (op
& 0x20) ? " +flag5?" : "",
382 (op
& 0x40) ? 'F' : 'T',
383 op
& 3, method_base
[op
& 3]);
385 if ((op
& 0x50) != 0x50) {
386 printf(" index 0x%04x", get_index(&p
));
390 /* FIXUP subrecord */
393 printf(" FIXUP %s-rel location %2d offset 0x%03x",
394 (op
& 0x40) ? "seg" : "self",
396 ((op
& 3) << 8) + *p
++);
399 printf("\n frame %s%d%s",
400 (fix
& 0x80) ? "thread " : "F",
402 ((fix
& 0xc0) == 0xc0) ? "?" : "");
404 if ((fix
& 0xc0) == 0)
405 printf(" datum 0x%04x", get_index(&p
));
407 printf("\n target %s%d",
408 (fix
& 0x10) ? "thread " : "method T",
411 if ((fix
& 0x10) == 0)
412 printf(" (%s)", method_base
[fix
& 3]);
414 printf(" datum 0x%04x", get_index(&p
));
416 if ((fix
& 0x08) == 0) {
418 printf(" disp 0x%08x", get_32(&p
));
420 printf(" disp 0x%04x", get_16(&p
));
425 hexdump_data(start
-data
, start
, p
-start
, n
-(start
-data
));
429 static const dump_func dump_type
[256] =
431 [0x88] = dump_coment
,
432 [0x96] = dump_lnames
,
433 [0x98] = dump_segdef
,
434 [0x99] = dump_segdef
,
435 [0x9c] = dump_fixupp
,
436 [0x9d] = dump_fixupp
,
437 [0xca] = dump_lnames
,
445 const uint8_t *p
, *data
;
452 data
= mmap(NULL
, len
, PROT_READ
, MAP_PRIVATE
, fd
, 0);
453 if (data
== MAP_FAILED
)
462 n
= *(uint16_t *)(p
+1);
464 printf("%02x %-10s %4zd bytes",
466 record_types
[type
] ? record_types
[type
] : "???",
470 printf("\n (truncated, only %zd bytes left)\n", len
-3);
471 break; /* Truncated */
474 p
+= 3; /* Header doesn't count in the length */
475 n
--; /* Remove checksum byte */
478 for (i
= -3; i
< (int)n
; i
++)
481 printf(", checksum %02X", p
[i
]);
483 printf(" (valid)\n");
485 printf(" (actual = %02X)\n", csum
);
488 dump_type
[type
](type
, p
, n
);
490 dump_unknown(type
, p
, n
);
496 munmap((void *)data
, st
.st_size
);
500 int main(int argc
, char *argv
[])
507 for (i
= 1; i
< argc
; i
++) {
508 fd
= open(argv
[i
], O_RDONLY
);
509 if (fd
< 0 || dump_omf(fd
)) {