Compare commits

..

No commits in common. "544f01f38089192f7b8047ce20222799d2efe16f" and "5b0cd41383ec274d1c48638f6d0898ef7156ae5a" have entirely different histories.

17 changed files with 63 additions and 605 deletions

1
.gitignore vendored
View file

@ -9,4 +9,3 @@
/debug_bld
/out
/ft_nm
/ft_otool

70
.vscode/launch.json vendored
View file

@ -5,7 +5,7 @@
"version": "0.2.0",
"configurations": [
{
"name": "nm self read",
"name": "self read",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/debug_bld/ft_nm",
@ -21,7 +21,7 @@
}
},
{
"name": "nm orig nm read",
"name": "orig nm read",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/debug_bld/ft_nm",
@ -35,70 +35,6 @@
"osx": {
"MIMode": "lldb"
}
},
{
"name": "nm 32bin",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/debug_bld/ft_nm",
"args": ["t/test_binaries/test32"],
"preLaunchTask": "build",
"stopAtEntry": false,
"cwd": "${workspaceFolder}",
"environment": [],
"externalConsole": false,
"MIMode": "gdb",
"osx": {
"MIMode": "lldb"
}
},
{
"name": "nm fat",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/debug_bld/ft_nm",
"args": ["/usr/bin/javah"],
"preLaunchTask": "build",
"stopAtEntry": false,
"cwd": "${workspaceFolder}",
"environment": [],
"externalConsole": false,
"MIMode": "gdb",
"osx": {
"MIMode": "lldb"
}
},
{
"name": "nm ar",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/debug_bld/ft_nm",
"args": ["${workspaceFolder}/debug_bld/subprojects/libft/libft.a"],
"preLaunchTask": "build",
"stopAtEntry": false,
"cwd": "${workspaceFolder}",
"environment": [],
"externalConsole": false,
"MIMode": "gdb",
"osx": {
"MIMode": "lldb"
}
},
{
"name": "otool orig nm read",
"type": "cppdbg",
"request": "launch",
"program": "${workspaceFolder}/debug_bld/ft_otool",
"args": ["/usr/bin/nm"],
"preLaunchTask": "build",
"stopAtEntry": false,
"cwd": "${workspaceFolder}",
"environment": [],
"externalConsole": false,
"MIMode": "gdb",
"osx": {
"MIMode": "lldb"
}
},
}
]
}

View file

@ -5,15 +5,13 @@ OUT_DIR="$PWD/out"
RELEASE_BLD_DIR="$PWD/bld"
DEBUG_BLD_DIR="$PWD/debug_bld"
FT_NM="ft_nm"
FT_OTOOL="ft_otool"
TARGET="ft_nm"
meson $RELEASE_BLD_DIR --prefix=$OUT_DIR --libdir=$OUT_DIR --includedir=$OUT_DIR --bindir=$OUT_DIR --buildtype=release
meson $RELEASE_BLD_DIR --prefix=$OUT_DIR --libdir=$OUT_DIR --includedir=$OUT_DIR --buildtype=release
meson $DEBUG_BLD_DIR --buildtype=debug
ninja -C $DEBUG_BLD_DIR
ninja -C $RELEASE_BLD_DIR
ninja -C $RELEASE_BLD_DIR install
ln -sf "$OUT_DIR//$FT_NM" "$PWD/$FT_NM"
ln -sf "$OUT_DIR/$FT_OTOOL" "$PWD/$FT_OTOOL"
ln -sf "$OUT_DIR/bin/$TARGET" "$PWD/$TARGET"

View file

@ -1,28 +0,0 @@
#ifndef FT_FILE
# define FT_FILE
typedef struct s_file
{
void *file;
uint32_t size;
int fd;
} t_file;
void init_file
(
t_file *file
);
void open_file
(
const char *filename,
t_file *file
);
void close_file
(
t_file *file
);
#endif

View file

@ -6,7 +6,7 @@
/* By: gtertysh <gtertysh@student.unit.ua> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2019/05/06 22:37:16 by foton #+# #+# */
/* Updated: 2019/08/17 14:26:16 by gtertysh ### ########.fr */
/* Updated: 2019/07/20 15:38:47 by gtertysh ### ########.fr */
/* */
/* ************************************************************************** */
@ -15,63 +15,51 @@
#include <mach-o/loader.h>
#include <mach-o/nlist.h>
#include <mach-o/fat.h>
#include <ar.h>
#include <stddef.h>
#include "ft_file.h"
typedef struct symtab_command t_symtab_command;
typedef struct load_command t_load_command;
typedef struct mach_header_64 t_mach_header_64;
typedef struct mach_header t_mach_header_32;
typedef struct symtab_command t_symtab_command;
typedef struct segment_command_64 t_segment_command_64;
typedef struct segment_command t_segment_command_32;
typedef struct nlist_64 t_nlist_64;
typedef struct nlist t_nlist_32;
typedef struct section_64 t_section_64;
typedef struct section t_section_32;
typedef struct s_nm_file
{
void *file;
uint32_t size;
int fd;
} t_nm_file;
typedef struct s_nm_mach_64
{
t_mach_header_64 *header;
t_load_command *commands;
t_symtab_command *symbol_table_command;
t_segment_command_64 *text_command;
t_section_64 *text_section;
t_nlist_64 *symbol_table;
char *string_table;
} t_nm_mach_64;
typedef struct s_nm_mach_32
{
t_mach_header_32 *header;
t_load_command *commands;
t_symtab_command *symbol_table_command;
t_segment_command_32 *text_command;
t_section_32 *text_section;
t_nlist_32 *symbol_table;
char *string_table;
} t_nm_mach_32;
void fat
void init_file
(
t_file *file
t_nm_file *file
);
void ar
void open_file
(
t_file *file
const char *filename,
t_nm_file *file
);
void close_file
(
t_nm_file *file
);
void macho64
(
t_file *file
);
void macho32
(
t_file *file
t_nm_file *file
);
t_symtab_command *find_symbol_table_command
@ -82,8 +70,9 @@ t_symtab_command *find_symbol_table_command
void print_addr
(
size_t addr,
size_t count
void *addr
);
#endif

View file

@ -25,41 +25,27 @@ add_global_arguments(
inc = include_directories('inc')
# install_headers('inc/ft_nm.h')
install_headers('inc/ft_nm.h')
nm_sources = [
'src/nm/main.c',
'src/nm/nm_file.c',
'src/nm/macho64.c',
'src/nm/macho32.c',
'src/nm/fat.c',
'src/nm/ar.c',
'src/nm/print_address.c',
'src/nm/find_symbol_table_command.c',
sources = [
'src/main.c',
'src/nm_file.c',
'src/macho64.c',
'src/print_address.c',
'src/find_symbol_table_command.c',
]
otool_sources = [
'src/nm/nm_file.c',
'src/nm/print_address.c',
'src/otool/main.c'
]
libft = subproject('libft')
libft_dep = libft.get_variable('libft_dep')
libft_dep = subproject('libft').get_variable('libft_dep')
ft_nm = executable(
'ft_nm',
nm_sources,
sources,
include_directories: inc,
dependencies: libft_dep,
install: true,
)
ft_nm = executable(
'ft_otool',
otool_sources,
include_directories: inc,
dependencies: libft_dep,
install: true,
)
#subdir('t')

View file

@ -2,7 +2,7 @@
#include "libft.h"
#include <stddef.h>
static t_section_64 *find_section
t_section_64 *find_section
(
t_nm_mach_64 *mach64,
uint32_t index
@ -36,14 +36,13 @@ static t_section_64 *find_section
return NULL;
}
static void print_symbol_table(t_nm_mach_64 *mach64)
void print_symbol_table(t_nm_mach_64 *mach64)
{
uint32_t j;
t_nlist_64 symbol;
t_section_64 *section;
j = 0;
while(j < mach64->symbol_table_command->nsyms)
{
symbol = mach64->symbol_table[j];
@ -62,7 +61,7 @@ static void print_symbol_table(t_nm_mach_64 *mach64)
// some shit herustic should be used
// to determine to print address or not
if (symbol.n_value)
print_addr(symbol.n_value, 16);
print_addr((void *)symbol.n_value);
else
ft_putstr(" ");
ft_putchar(' ');
@ -72,6 +71,7 @@ static void print_symbol_table(t_nm_mach_64 *mach64)
ft_putchar('A' + offset);
if (type == N_SECT)
{
// lookup in which section symbol is located
section = find_section(mach64, symbol.n_sect);
if(ft_strcmp(SECT_TEXT, section->sectname) == 0)
ft_putchar('T' + offset);
@ -89,7 +89,7 @@ static void print_symbol_table(t_nm_mach_64 *mach64)
}
}
void macho64(t_file *file)
void macho64(t_nm_file *file)
{
t_nm_mach_64 mach64;

View file

@ -6,7 +6,7 @@
/* By: gtertysh <gtertysh@student.unit.ua> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */
/* Created: 2019/07/13 14:52:27 by gtertysh #+# #+# */
/* Updated: 2019/08/10 16:05:21 by gtertysh ### ########.fr */
/* Updated: 2019/07/20 15:21:14 by gtertysh ### ########.fr */
/* */
/* ************************************************************************** */
@ -16,51 +16,34 @@
void hanldle_file(const char *filename)
{
t_file file;
t_nm_file file;
uint32_t magic;
// create dispatch table for this bullshit
init_file(&file);
open_file(filename, &file);
magic = *(uint32_t *)file.file;
if (magic == MH_MAGIC_64)
macho64(&file);
else if (magic == MH_MAGIC)
macho32(&file);
else if (magic == FAT_CIGAM)
fat(&file);
else if (ft_strncmp(file.file, ARMAG, SARMAG) == 0)
ar(&file);
// else if (magic == MH_MAGIC)
// handle_32(file);
// else if (magic == MH_CIGAM || magic == MH_CIGAM_64)
// ft_putstr("do not support big endian binaries.");
else
ft_putstr("invalid magic number.");
close_file(&file);
}
void print_name(const char *name)
{
ft_putstr("\n");
ft_putstr(name);
ft_putstr(":\n");
}
int main(int argc, char **argv)
{
int i;
if (argc == 1)
hanldle_file("a.out");
else if (argc == 2)
hanldle_file(argv[1]);
else
{
i = 1;
while (i < argc)
{
print_name(argv[i]);
hanldle_file(argv[i]);
i++;
}
hanldle_file(argv[i++]);
}
return (0);
}

View file

@ -1,31 +0,0 @@
#include "ft_nm.h"
#include "libft.h"
#include <ar.h>
void ar(t_file *file)
{
struct ar_hdr *runner;
char *long_name;
size_t offset;
t_file obj;
runner = file->file + SARMAG;
while ((void *)runner < file->file + file->size)
{
if (ft_strncmp(runner->ar_name, AR_EFMT1, 3) == 0)
{
offset = ft_atoi(&runner->ar_name[3]);
long_name = (char *)runner + sizeof(struct ar_hdr);
ft_putstr("\n");
ft_putstr(long_name);
ft_putstr("\n");
obj.file = (void *)((size_t)runner + sizeof(struct ar_hdr) + offset);
obj.size = ft_atoi(runner->ar_size) - offset;
if (*(uint32_t *)obj.file == MH_MAGIC_64)
macho64(&obj);
}
runner = (struct ar_hdr *)((size_t)runner + ft_atoi(runner->ar_size) + sizeof(struct ar_hdr));
}
}

View file

@ -1,33 +0,0 @@
#include "ft_nm.h"
#include "libft.h"
#include <stdio.h>
void fat(t_file *file)
{
struct fat_arch *arch_runner;
struct fat_header *header;
t_file arch_file;
uint32_t i;
header = (struct fat_header *)file->file;
if (header->magic != FAT_CIGAM)
{
ft_putstr_fd("only 32bit big endian fat binaries supported\n",
STDERR_FILENO);
exit(1);
}
arch_runner = file->file + sizeof(struct fat_header);
i = 0;
while (i < OSSwapInt32(header->nfat_arch))
{
if (OSSwapInt32(arch_runner->cputype) == CPU_TYPE_X86_64)
{
arch_file.file = file->file + OSSwapInt32(arch_runner->offset);
arch_file.size = OSSwapInt32(arch_runner->size);
macho64(&arch_file);
}
arch_runner = arch_runner + sizeof(struct fat_arch);
i++;
}
}

View file

@ -1,105 +0,0 @@
#include "ft_nm.h"
#include "libft.h"
#include <stddef.h>
static t_section_32 *find_section
(
t_nm_mach_32 *mach32,
uint32_t index
)
{
uint32_t i;
uint32_t acc;
t_load_command *lc;
t_section_32 *sections;
t_segment_command_32 *segment;
lc = mach32->commands;
acc = 0;
i = 0;
while(i < mach32->header->ncmds)
{
if (lc->cmd == LC_SEGMENT)
{
segment = (t_segment_command_32 *)lc;
if (segment->nsects + acc >= index)
{
sections = (t_section_32 *)((uintptr_t)segment +
sizeof(t_segment_command_32));
return (sections + index - acc - 1);
}
acc += segment->nsects;
}
lc = (t_load_command *)((uintptr_t)lc + lc->cmdsize);
i++;
}
return NULL;
}
static void print_symbol_table(t_nm_mach_32 *mach32)
{
uint32_t j;
t_nlist_32 symbol;
t_section_32 *section;
j = 0;
while(j < mach32->symbol_table_command->nsyms)
{
symbol = mach32->symbol_table[j];
int type = symbol.n_type & N_TYPE;
int external = symbol.n_type & N_EXT;
int debug = (symbol.n_type & N_STAB) != 0;
int offset = external ? 0 : 32;
char *name = mach32->string_table + symbol.n_un.n_strx;
if (debug)
{
j++;
continue;
}
// some shit herustic should be used
// to determine to print address or not
if (symbol.n_value)
print_addr(symbol.n_value, 8);
else
ft_putstr(" ");
ft_putchar(' ');
if (type == N_UNDF)
ft_putchar('U' + offset);
if (type == N_ABS)
ft_putchar('A' + offset);
if (type == N_SECT)
{
section = find_section(mach32, symbol.n_sect);
if(ft_strcmp(SECT_TEXT, section->sectname) == 0)
ft_putchar('T' + offset);
else if(ft_strcmp(SECT_DATA, section->sectname) == 0)
ft_putchar('D' + offset);
else if(ft_strcmp(SECT_BSS, section->sectname) == 0)
ft_putchar('B' + offset);
else
ft_putchar('S' + offset);
}
ft_putchar(' ');
ft_putstr(name);
ft_putstr("\n");
j++;
}
}
void macho32(t_file *file)
{
t_nm_mach_32 mach32;
mach32.header = (t_mach_header_32 *)file->file;
mach32.commands = (t_load_command *)(file->file + sizeof(t_mach_header_32));
mach32.symbol_table_command = find_symbol_table_command(mach32.commands,
mach32.header->ncmds);
mach32.symbol_table = (t_nlist_32 *)
(mach32.symbol_table_command->symoff + file->file);
mach32.string_table = (char *)
(mach32.symbol_table_command->stroff + file->file);
print_symbol_table(&mach32);
}

View file

@ -8,27 +8,27 @@
#include <sys/mman.h>
#include <stdio.h>
void init_file(t_file *file)
void init_file(t_nm_file *file)
{
file->fd = -1;
file->file = NULL;
file->size = 0;
}
void open_file(const char *filename, t_file *file)
void open_file(const char *filename, t_nm_file *file)
{
struct stat stat_buff;
file->fd = open(filename, O_RDONLY);
if (file->fd == -1)
{
perror(filename);
ft_putstr("can't open file\n");
exit(1);
}
stat(filename, &stat_buff);
if (!S_ISREG(stat_buff.st_mode))
{
ft_putstr_fd("not a regular file\n", STDERR_FILENO);
ft_putstr("not a regular file\n");
close(file->fd);
exit(1);
}
@ -42,7 +42,7 @@ void open_file(const char *filename, t_file *file)
}
}
void close_file(t_file *file)
void close_file(t_nm_file *file)
{
close(file->fd);
munmap(file->file, file->size);

View file

@ -1,238 +0,0 @@
#include <mach-o/loader.h>
#include <mach-o/fat.h>
#include <ar.h>
#include "ft_file.h"
#include "ft_nm.h"
#include "libft.h"
t_segment_command_64 *find_text_command_64
(
t_load_command *lc,
uint32_t count
)
{
uint32_t i;
t_segment_command_64 *text_command;
text_command = NULL;
i = 0;
while(i < count)
{
if (lc->cmd == LC_SEGMENT_64 &&
ft_strcmp(((t_segment_command_64 *)lc)->segname, SEG_TEXT) == 0)
{
text_command = (t_segment_command_64 *)lc;
break;
}
lc = (t_load_command *)((uintptr_t)lc + lc->cmdsize);
i++;
}
return text_command;
}
t_segment_command_32 *find_text_command_32
(
t_load_command *lc,
uint32_t count
)
{
uint32_t i;
t_segment_command_32 *text_command;
text_command = NULL;
i = 0;
while(i < count)
{
if (lc->cmd == LC_SEGMENT &&
ft_strcmp(((t_segment_command_32 *)lc)->segname, SEG_TEXT) == 0)
{
text_command = (t_segment_command_32 *)lc;
break;
}
lc = (t_load_command *)((uintptr_t)lc + lc->cmdsize);
i++;
}
return text_command;
}
t_section_64 *find_text_section_64
(
t_segment_command_64 *text_segment
)
{
uint32_t i;
t_section_64 *text_section;
t_section_64 *runner;
i = 0;
text_section = NULL;
runner = (t_section_64 *)((uintptr_t)text_segment +
sizeof(t_segment_command_64));
while(i < text_segment->nsects)
{
if (ft_strcmp(runner->sectname, SECT_TEXT) == 0)
{
text_section = runner;
break;
}
runner = (t_section_64 *)((uintptr_t)runner + sizeof(t_section_64));
i++;
}
return (text_section);
}
t_section_32 *find_text_section_32
(
t_segment_command_32 *text_segment
)
{
uint32_t i;
t_section_32 *text_section;
t_section_32 *runner;
i = 0;
text_section = NULL;
runner = (t_section_32 *)((uintptr_t)text_segment +
sizeof(t_segment_command_32));
while(i < text_segment->nsects)
{
if (ft_strcmp(runner->sectname, SECT_TEXT) == 0)
{
text_section = runner;
break;
}
runner = (t_section_32 *)((uintptr_t)runner + sizeof(t_section_32));
i++;
}
return (text_section);
}
void print_text_64
(
t_nm_mach_64 *mach64,
t_file *file
)
{
size_t text;
size_t addr;
uint64_t i;
uint64_t j;
text = (size_t)file->file + mach64->text_section->offset;
addr = mach64->text_section->addr;
i = 0;
while(i < mach64->text_section->size)
{
print_addr(addr + i, 16);
ft_putstr("\t");
j = 0;
while(i < mach64->text_section->size && j < 16)
{
print_addr(*(char *)(text + i), 2);
ft_putstr(" ");
j++;
i++;
}
ft_putstr("\n");
}
}
void print_text_32
(
t_nm_mach_32 *mach32,
t_file *file
)
{
size_t text;
size_t addr;
uint64_t i;
uint64_t j;
text = (size_t)file->file + mach32->text_section->offset;
addr = mach32->text_section->addr;
i = 0;
while(i < mach32->text_section->size)
{
print_addr(addr + i, 8);
ft_putstr("\t");
j = 0;
while(i < mach32->text_section->size && j < 8)
{
print_addr(*(char *)(text + i), 2);
ft_putstr(" ");
j++;
i++;
}
ft_putstr("\n");
}
}
void otool64(t_file *file)
{
t_nm_mach_64 mach64;
mach64.header = (t_mach_header_64 *)file->file;
mach64.commands = (t_load_command *)(file->file + sizeof(t_mach_header_64));
mach64.text_command = find_text_command_64(mach64.commands,
mach64.header->ncmds);
mach64.text_section = find_text_section_64(mach64.text_command);
print_text_64(&mach64, file);
}
void otool32(t_file *file)
{
t_nm_mach_32 mach32;
mach32.header = (t_mach_header_32 *)file->file;
mach32.commands = (t_load_command *)(file->file + sizeof(t_mach_header_32));
mach32.text_command = find_text_command_32(mach32.commands,
mach32.header->ncmds);
mach32.text_section = find_text_section_32(mach32.text_command);
print_text_32(&mach32, file);
}
void hanldle_file(const char *filename)
{
t_file file;
uint32_t magic;
// create dispatch table for this bullshit
init_file(&file);
open_file(filename, &file);
magic = *(uint32_t *)file.file;
if (magic == MH_MAGIC_64)
otool64(&file);
else if (magic == MH_MAGIC)
otool32(&file);
// else if (magic == FAT_CIGAM)
// fat(&file);
// else if (ft_strncmp(file.file, ARMAG, SARMAG) == 0)
// ar(&file);
else
ft_putstr("invalid magic number.");
close_file(&file);
}
int main(int argc, char **argv)
{
int i;
if (argc == 1)
{
ft_putstr_fd("at least one file must be specified\n", STDERR_FILENO);
return (1);
}
i = 1;
while(i < argc)
{
ft_putstr(argv[i]);
ft_putstr(":\n");
ft_putstr("Contents of (__TEXT,__text) section\n");
hanldle_file(argv[i]);
i++;
}
return (0);
}

View file

@ -2,15 +2,17 @@
#include "libft.h"
#include <stddef.h>
void print_addr(size_t addr, size_t count)
void print_addr(void *addr)
{
size_t i;
size_t bit_4;
size_t count;
count = sizeof(addr) * 2;
i = 0;
while (i < count)
{
bit_4 = (addr >> ((count - i - 1) * 4)) & 0xf;
bit_4 = ((size_t)addr >> ((count - i - 1) * 4)) & 0xf;
if (bit_4 < 10)
ft_putnbr(bit_4);
else

Binary file not shown.

Binary file not shown.