static char *default_outputfile(TCCState *s, const char *first_file) { char buf[1024]; char *ext; const char *name = "a"; if (first_file && strcmp(first_file, "-")) name = tcc_basename(first_file); pstrcpy(buf, sizeof(buf), name); ext = tcc_fileextension(buf); #ifdef TCC_TARGET_PE if (s->output_type == TCC_OUTPUT_DLL) strcpy(ext, ".dll"); else if (s->output_type == TCC_OUTPUT_EXE) strcpy(ext, ".exe"); else #endif if (( (s->output_type == TCC_OUTPUT_OBJ && !s->option_r) || (s->output_type == TCC_OUTPUT_PREPROCESS) ) && *ext) strcpy(ext, ".o"); else strcpy(buf, "a.out"); return tcc_strdup(buf); }
static void exec_other_tcc(TCCState *s, char **argv, const char *optarg) { char child_path[4096], *child_name; const char *target; switch (atoi(optarg)) { #ifdef TCC_TARGET_I386 case 32: break; case 64: target = "x86_64"; #else case 64: break; case 32: target = "i386"; #endif pstrcpy(child_path, sizeof child_path - 40, argv[0]); child_name = tcc_basename(child_path); strcpy(child_name, target); #ifdef TCC_TARGET_PE strcat(child_name, "-win32"); #endif strcat(child_name, "-tcc"); if (strcmp(argv[0], child_path)) { if (s->verbose > 0) printf("tcc: using '%s'\n", child_name), fflush(stdout); execvp(argv[0] = child_path, argv); } tcc_error("'%s' not found", child_name); case 0: /* ignore -march etc. */ break; default: tcc_warning("unsupported option \"-m%s\"", optarg); } }
static void pe_build_exports(struct pe_info *pe) { Elf32_Sym *sym; int sym_index, sym_end; DWORD rva_base, func_o, name_o, ord_o, str_o; struct pe_export_header *hdr; int sym_count, n, ord, *sorted, *sp; FILE *op; char buf[260]; const char *dllname; const char *name; rva_base = pe->thunk->sh_addr - pe->imagebase; sym_count = 0; n = 1; sorted = NULL; op = NULL; sym_end = symtab_section->data_offset / sizeof(Elf32_Sym); for (sym_index = 1; sym_index < sym_end; ++sym_index) { sym = (Elf32_Sym *) symtab_section->data + sym_index; name = symtab_section->link->data + sym->st_name; // Only export symbols from actually written sections if ((sym->st_other & 1) && pe->s1->sections[sym->st_shndx]->sh_addr) { dynarray_add((void ***) &sorted, &sym_count, (void *) n); dynarray_add((void ***) &sorted, &sym_count, (void *) name); } ++n; } if (sym_count == 0) return; sym_count /= 2; qsort(sorted, sym_count, 2 * sizeof(sorted[0]), sym_cmp); pe_align_section(pe->thunk, 16); dllname = tcc_basename(pe->filename); pe->exp_offs = pe->thunk->data_offset; func_o = pe->exp_offs + sizeof(struct pe_export_header); name_o = func_o + sym_count * sizeof(DWORD); ord_o = name_o + sym_count * sizeof(DWORD); str_o = ord_o + sym_count * sizeof(WORD); hdr = section_ptr_add(pe->thunk, str_o - pe->exp_offs); hdr->Characteristics = 0; hdr->Base = 1; hdr->NumberOfFunctions = sym_count; hdr->NumberOfNames = sym_count; hdr->AddressOfFunctions = func_o + rva_base; hdr->AddressOfNames = name_o + rva_base; hdr->AddressOfNameOrdinals = ord_o + rva_base; hdr->Name = str_o + rva_base; put_elf_str(pe->thunk, dllname); if (pe->def != NULL) { // Write exports to .def file op = fopen(pe->def, "w"); if (op == NULL) { error_noabort("could not create '%s': %s", pe->def, strerror(errno)); } else { fprintf(op, "LIBRARY %s\n\nEXPORTS\n", dllname); if (verbose) { printf("<- %s (%d symbols)\n", buf, sym_count); } } } for (sp = sorted, ord = 0; ord < sym_count; ++ord, sp += 2) { sym_index = sp[0]; name = (const char *) sp[1]; // Insert actual address later in pe_relocate_rva put_elf_reloc(symtab_section, pe->thunk, func_o, R_386_RELATIVE, sym_index); *(DWORD *)(pe->thunk->data + name_o) = pe->thunk->data_offset + rva_base; *(WORD *)(pe->thunk->data + ord_o) = ord; put_elf_str(pe->thunk, name); func_o += sizeof(DWORD); name_o += sizeof(DWORD); ord_o += sizeof(WORD); if (op) fprintf(op, "%s@%d\n", name, ord); } pe->exp_size = pe->thunk->data_offset - pe->exp_offs; tcc_free(sorted); if (op) fclose(op); }
ST_FN void pe_build_exports(struct pe_info *pe) { Elf32_Sym *sym; int sym_index, sym_end; DWORD rva_base, func_o, name_o, ord_o, str_o; struct pe_export_header *hdr; int sym_count, n, ord, *sorted, *sp; FILE *op; char buf[MAX_PATH]; const char *dllname; const char *name; rva_base = pe->thunk->sh_addr - pe->imagebase; sym_count = 0, n = 1, sorted = NULL, op = NULL; sym_end = symtab_section->data_offset / sizeof(Elf32_Sym); for (sym_index = 1; sym_index < sym_end; ++sym_index) { sym = (Elf32_Sym*)symtab_section->data + sym_index; name = symtab_section->link->data + sym->st_name; if ((sym->st_other & 1) /* export only symbols from actually written sections */ && pe->s1->sections[sym->st_shndx]->sh_addr) { dynarray_add((void***)&sorted, &sym_count, (void*)n); dynarray_add((void***)&sorted, &sym_count, (void*)name); } ++n; #if 0 if (sym->st_other & 1) printf("export: %s\n", name); if (sym->st_other & 2) printf("stdcall: %s\n", name); #endif } if (0 == sym_count) return; sym_count /= 2; qsort (sorted, sym_count, 2 * sizeof sorted[0], sym_cmp); pe_align_section(pe->thunk, 16); dllname = tcc_basename(pe->filename); pe->exp_offs = pe->thunk->data_offset; func_o = pe->exp_offs + sizeof(struct pe_export_header); name_o = func_o + sym_count * sizeof (DWORD); ord_o = name_o + sym_count * sizeof (DWORD); str_o = ord_o + sym_count * sizeof(WORD); hdr = section_ptr_add(pe->thunk, str_o - pe->exp_offs); hdr->Characteristics = 0; hdr->Base = 1; hdr->NumberOfFunctions = sym_count; hdr->NumberOfNames = sym_count; hdr->AddressOfFunctions = func_o + rva_base; hdr->AddressOfNames = name_o + rva_base; hdr->AddressOfNameOrdinals = ord_o + rva_base; hdr->Name = str_o + rva_base; put_elf_str(pe->thunk, dllname); #if 1 /* automatically write exports to <output-filename>.def */ strcpy(buf, pe->filename); strcpy(tcc_fileextension(buf), ".def"); op = fopen(buf, "w"); if (NULL == op) { error_noabort("could not create '%s': %s", buf, strerror(errno)); } else { fprintf(op, "LIBRARY %s\n\nEXPORTS\n", dllname); if (pe->s1->verbose) printf("<- %s (%d symbols)\n", buf, sym_count); } #endif for (sp = sorted, ord = 0; ord < sym_count; ++ord, sp += 2) { sym_index = sp[0], name = (const char *)sp[1]; /* insert actual address later in pe_relocate_rva */ put_elf_reloc(symtab_section, pe->thunk, func_o, R_386_RELATIVE, sym_index); *(DWORD*)(pe->thunk->data + name_o) = pe->thunk->data_offset + rva_base; *(WORD*)(pe->thunk->data + ord_o) = ord; put_elf_str(pe->thunk, name); func_o += sizeof (DWORD); name_o += sizeof (DWORD); ord_o += sizeof (WORD); if (op) fprintf(op, "%s\n", name); } pe->exp_size = pe->thunk->data_offset - pe->exp_offs; tcc_free(sorted); }
// Extract the file extension of a file name char *tcc_fileextension(const char *name) { char *b = tcc_basename(name); char *e = strrchr(b, '.'); return e ? e : strchr(b, 0); }