1 /* grub-pe2elf.c - tool to convert pe image to elf. */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2008 Free Software Foundation, Inc.
6 * GRUB is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * GRUB is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
21 #include <grub/types.h>
22 #include <grub/util/misc.h>
24 #include <grub/efi/pe32.h>
32 static struct option options
[] = {
33 {"help", no_argument
, 0, 'h'},
34 {"version", no_argument
, 0, 'V'},
35 {"verbose", no_argument
, 0, 'v'},
43 fprintf (stderr
, "Try ``grub-editenv --help'' for more information.\n");
46 Usage: grub-editenv [OPTIONS] input [output]\n\
48 Tool to convert pe image to elf.\n\
50 -h, --help display this message and exit\n\
51 -V, --version print version information and exit\n\
52 -v, --verbose print verbose messages\n\
54 Report bugs to <%s>.\n", PACKAGE_BUGREPORT
);
74 #define TEXT_SECTION 1
75 #define RDATA_SECTION 2
76 #define DATA_SECTION 3
78 #define MODNAME_SECTION 5
79 #define MODDEPS_SECTION 6
80 #define SYMTAB_SECTION 7
81 #define STRTAB_SECTION 8
84 #define MAX_SECTIONS 12
86 #define STRTAB_BLOCK 256
89 static int strtab_max
, strtab_len
;
92 Elf32_Shdr shdr
[MAX_SECTIONS
];
97 insert_string (char *name
)
105 if (strtab_len
+ len
>= strtab_max
)
107 strtab_max
+= STRTAB_BLOCK
;
108 strtab
= xrealloc (strtab
, strtab_max
);
111 strcpy (strtab
+ strtab_len
, name
);
113 strtab_len
+= len
+ 1;
119 write_section_data (FILE* fp
, char *image
,
120 struct grub_pe32_coff_header
*pe_chdr
,
121 struct grub_pe32_section_table
*pe_shdr
)
126 section_map
= xmalloc ((pe_chdr
->num_sections
+ 1) * sizeof (int));
129 for (i
= 0; i
< pe_chdr
->num_sections
; i
++, pe_shdr
++)
133 if (! strcmp (pe_shdr
->name
, ".text"))
136 shdr
[idx
].sh_flags
= SHF_ALLOC
| SHF_EXECINSTR
;
138 else if (! strcmp (pe_shdr
->name
, ".rdata"))
141 shdr
[idx
].sh_flags
= SHF_ALLOC
;
143 else if (! strcmp (pe_shdr
->name
, ".data"))
146 shdr
[idx
].sh_flags
= SHF_ALLOC
| SHF_WRITE
;
148 else if (! strcmp (pe_shdr
->name
, ".bss"))
151 shdr
[idx
].sh_flags
= SHF_ALLOC
| SHF_WRITE
;
153 else if (! strcmp (pe_shdr
->name
, ".modname"))
154 idx
= MODNAME_SECTION
;
155 else if (! strcmp (pe_shdr
->name
, ".moddeps"))
156 idx
= MODDEPS_SECTION
;
159 section_map
[i
+ 1] = -1;
163 section_map
[i
+ 1] = idx
;
165 shdr
[idx
].sh_type
= (idx
== BSS_SECTION
) ? SHT_NOBITS
: SHT_PROGBITS
;
166 shdr
[idx
].sh_size
= pe_shdr
->raw_data_size
;
167 shdr
[idx
].sh_addralign
= 1 << (((pe_shdr
->characteristics
>>
168 GRUB_PE32_SCN_ALIGN_SHIFT
) &
169 GRUB_PE32_SCN_ALIGN_MASK
) - 1);
171 if (idx
!= BSS_SECTION
)
173 shdr
[idx
].sh_offset
= offset
;
174 grub_util_write_image_at (image
+ pe_shdr
->raw_data_offset
,
175 pe_shdr
->raw_data_size
, offset
, fp
);
177 offset
+= pe_shdr
->raw_data_size
;
180 if (pe_shdr
->relocations_offset
)
182 char name
[5 + strlen (pe_shdr
->name
)];
184 if (num_sections
>= MAX_SECTIONS
)
185 grub_util_error ("Too many sections");
187 sprintf (name
, ".rel%s", pe_shdr
->name
);
189 shdr
[num_sections
].sh_name
= insert_string (name
);
190 shdr
[num_sections
].sh_link
= i
;
191 shdr
[num_sections
].sh_info
= idx
;
193 shdr
[idx
].sh_name
= shdr
[num_sections
].sh_name
+ 4;
198 shdr
[idx
].sh_name
= insert_string (pe_shdr
->name
);
205 write_reloc_section (FILE* fp
, char *image
,
206 struct grub_pe32_coff_header
*pe_chdr
,
207 struct grub_pe32_section_table
*pe_shdr
,
213 for (i
= REL_SECTION
; i
< num_sections
; i
++)
215 struct grub_pe32_section_table
*pe_sec
;
216 struct grub_pe32_reloc
*pe_rel
;
218 int num_rels
, j
, modified
;
220 pe_sec
= pe_shdr
+ shdr
[i
].sh_link
;
221 pe_rel
= (struct grub_pe32_reloc
*) (image
+ pe_sec
->relocations_offset
);
222 rel
= (Elf32_Rel
*) xmalloc (pe_sec
->num_relocations
* sizeof (Elf32_Rel
));
226 for (j
= 0; j
< pe_sec
->num_relocations
; j
++, pe_rel
++)
229 grub_uint32_t ofs
, *addr
;
231 if ((pe_rel
->symtab_index
>= pe_chdr
->num_symbols
) ||
232 (symtab_map
[pe_rel
->symtab_index
] == -1))
233 grub_util_error ("Invalid symbol");
235 if (pe_rel
->type
== GRUB_PE32_REL_I386_DIR32
)
237 else if (pe_rel
->type
== GRUB_PE32_REL_I386_REL32
)
240 grub_util_error ("Unknown pe relocation type %d\n", pe_rel
->type
);
242 ofs
= pe_rel
->offset
- pe_sec
->virtual_address
;
243 addr
= (grub_uint32_t
*)(image
+ pe_sec
->raw_data_offset
+ ofs
);
244 if (type
== R_386_PC32
)
248 code
= image
[pe_sec
->raw_data_offset
+ ofs
- 1];
250 if (((code
!= 0xe8) && (code
!= 0xe9)) || (*addr
))
251 grub_util_error ("Invalid relocation (%x %x)", code
, *addr
);
254 if (symtab
[symtab_map
[pe_rel
->symtab_index
]].st_shndx
)
256 if (symtab
[symtab_map
[pe_rel
->symtab_index
]].st_shndx
258 grub_util_error ("Cross section call is not allowed");
260 *addr
= (symtab
[symtab_map
[pe_rel
->symtab_index
]].st_value
269 rel
[num_rels
].r_offset
= ofs
;
270 rel
[num_rels
].r_info
= ELF32_R_INFO (symtab_map
[pe_rel
->symtab_index
],
276 grub_util_write_image_at (image
+ pe_sec
->raw_data_offset
,
277 shdr
[shdr
[i
].sh_info
].sh_size
,
278 shdr
[shdr
[i
].sh_info
].sh_offset
,
281 shdr
[i
].sh_type
= SHT_REL
;
282 shdr
[i
].sh_offset
= offset
;
283 shdr
[i
].sh_link
= SYMTAB_SECTION
;
284 shdr
[i
].sh_addralign
= 4;
285 shdr
[i
].sh_entsize
= sizeof (Elf32_Rel
);
286 shdr
[i
].sh_size
= num_rels
* sizeof (Elf32_Rel
);
288 grub_util_write_image_at (rel
, shdr
[i
].sh_size
, offset
, fp
);
289 offset
+= shdr
[i
].sh_size
;
295 write_symbol_table (FILE* fp
, char *image
,
296 struct grub_pe32_coff_header
*pe_chdr
,
297 struct grub_pe32_section_table
*pe_shdr
,
300 struct grub_pe32_symbol
*pe_symtab
;
303 int *symtab_map
, num_syms
;
306 pe_symtab
= (struct grub_pe32_symbol
*) (image
+ pe_chdr
->symtab_offset
);
307 pe_strtab
= (char *) (pe_symtab
+ pe_chdr
->num_symbols
);
309 symtab
= (Elf32_Sym
*) xmalloc ((pe_chdr
->num_symbols
+ 1) *
311 memset (symtab
, 0, (pe_chdr
->num_symbols
+ 1) * sizeof (Elf32_Sym
));
314 symtab_map
= (int *) xmalloc (pe_chdr
->num_symbols
* sizeof (int));
316 for (i
= 0; i
< (int) pe_chdr
->num_symbols
;
317 i
+= pe_symtab
->num_aux
+ 1, pe_symtab
+= pe_symtab
->num_aux
+ 1)
322 if ((pe_symtab
->section
> pe_chdr
->num_sections
) ||
323 (section_map
[pe_symtab
->section
] == -1))
326 if (! pe_symtab
->section
)
328 else if (pe_symtab
->type
== GRUB_PE32_DT_FUNCTION
)
333 if (pe_symtab
->storage_class
== GRUB_PE32_SYM_CLASS_EXTERNAL
)
338 if ((type
!= STT_FUNC
) && (pe_symtab
->num_aux
))
340 if (! pe_symtab
->value
)
343 symtab
[num_syms
].st_name
= shdr
[section_map
[pe_symtab
->section
]].sh_name
;
349 name
= ((pe_symtab
->long_name
[0]) ? pe_symtab
->short_name
:
350 pe_strtab
+ pe_symtab
->long_name
[1]);
352 if ((strcmp (name
, "_grub_mod_init")) &&
353 (strcmp (name
, "_grub_mod_fini")) &&
357 symtab
[num_syms
].st_name
= insert_string (name
);
360 symtab
[num_syms
].st_shndx
= section_map
[pe_symtab
->section
];
361 symtab
[num_syms
].st_value
= pe_symtab
->value
;
362 symtab
[num_syms
].st_info
= ELF32_ST_INFO (bind
, type
);
364 symtab_map
[i
] = num_syms
;
368 write_reloc_section (fp
, image
, pe_chdr
, pe_shdr
, symtab
, symtab_map
);
370 shdr
[SYMTAB_SECTION
].sh_name
= insert_string (".symtab");
371 shdr
[SYMTAB_SECTION
].sh_type
= SHT_SYMTAB
;
372 shdr
[SYMTAB_SECTION
].sh_offset
= offset
;
373 shdr
[SYMTAB_SECTION
].sh_size
= num_syms
* sizeof (Elf32_Sym
);
374 shdr
[SYMTAB_SECTION
].sh_entsize
= sizeof (Elf32_Sym
);
375 shdr
[SYMTAB_SECTION
].sh_link
= STRTAB_SECTION
;
376 shdr
[SYMTAB_SECTION
].sh_addralign
= 4;
378 grub_util_write_image_at (symtab
, shdr
[SYMTAB_SECTION
].sh_size
,
380 offset
+= shdr
[SYMTAB_SECTION
].sh_size
;
387 write_string_table (FILE* fp
)
389 shdr
[STRTAB_SECTION
].sh_name
= insert_string (".strtab");
390 shdr
[STRTAB_SECTION
].sh_type
= SHT_STRTAB
;
391 shdr
[STRTAB_SECTION
].sh_offset
= offset
;
392 shdr
[STRTAB_SECTION
].sh_size
= strtab_len
;
393 shdr
[STRTAB_SECTION
].sh_addralign
= 1;
394 grub_util_write_image_at (strtab
, strtab_len
, offset
, fp
);
395 offset
+= strtab_len
;
401 write_section_header (FILE* fp
)
403 ehdr
.e_ident
[EI_MAG0
] = ELFMAG0
;
404 ehdr
.e_ident
[EI_MAG1
] = ELFMAG1
;
405 ehdr
.e_ident
[EI_MAG2
] = ELFMAG2
;
406 ehdr
.e_ident
[EI_MAG3
] = ELFMAG3
;
407 ehdr
.e_ident
[EI_VERSION
] = EV_CURRENT
;
408 ehdr
.e_version
= EV_CURRENT
;
409 ehdr
.e_type
= ET_REL
;
411 ehdr
.e_ident
[EI_CLASS
] = ELFCLASS32
;
412 ehdr
.e_ident
[EI_DATA
] = ELFDATA2LSB
;
413 ehdr
.e_machine
= EM_386
;
415 ehdr
.e_ehsize
= sizeof (ehdr
);
416 ehdr
.e_shentsize
= sizeof (Elf32_Shdr
);
417 ehdr
.e_shstrndx
= STRTAB_SECTION
;
419 ehdr
.e_shoff
= offset
;
420 ehdr
.e_shnum
= num_sections
;
421 grub_util_write_image_at (&shdr
, sizeof (Elf32_Shdr
) * num_sections
,
424 grub_util_write_image_at (&ehdr
, sizeof (Elf32_Ehdr
), 0, fp
);
428 convert_pe (FILE* fp
, char *image
)
430 struct grub_pe32_coff_header
*pe_chdr
;
431 struct grub_pe32_section_table
*pe_shdr
;
434 pe_chdr
= (struct grub_pe32_coff_header
*) image
;
435 if (grub_le_to_cpu16 (pe_chdr
->machine
) != GRUB_PE32_MACHINE_I386
)
436 grub_util_error ("Invalid coff image");
438 strtab
= xmalloc (STRTAB_BLOCK
);
439 strtab_max
= STRTAB_BLOCK
;
443 offset
= sizeof (ehdr
);
444 pe_shdr
= (struct grub_pe32_section_table
*) (pe_chdr
+ 1);
445 num_sections
= REL_SECTION
;
447 section_map
= write_section_data (fp
, image
, pe_chdr
, pe_shdr
);
449 write_symbol_table (fp
, image
, pe_chdr
, pe_shdr
, section_map
);
452 write_string_table (fp
);
454 write_section_header (fp
);
458 main (int argc
, char *argv
[])
463 progname
= "grub-pe2elf";
465 /* Check for options. */
468 int c
= getopt_long (argc
, argv
, "hVv", options
, 0);
480 printf ("%s (%s) %s\n", progname
, PACKAGE_NAME
, PACKAGE_VERSION
);
496 fprintf (stderr
, "Filename not specified.\n");
500 image
= grub_util_read_image (argv
[optind
]);
502 if (optind
+ 1 < argc
)
505 fp
= fopen (argv
[optind
], "wb");
507 grub_util_error ("cannot open %s", argv
[optind
]);
509 convert_pe (fp
, image
);