Ragg2-cc -> Ragg2 merge (#9658)

- change of 'access' function for 'r_file_exists' for windows portability
- several changes in the way ragg2.c is done, see PR #9658 pancake comments
- change function for the remove of 'file'.text
- open changed for r_file_dump
- some elt of cEnv_t are now const
- skip all the pointers verification in set_cEnv but do them in check_cEnv instead
- add 2 r_str_sanitize for file and CC in parseCFile
- rewrite the removal of .o, .tmp, .bin, .s files, with cleaner code
- changed the long command with sed and grep to 2 C functions.
  - The function parseCompiled that basically does what the command was doing
  - And r_str_stripLines that is equivalent to "grep -v" (maybe we should put this one in str.c ?)
- simplify a bit getCompiler function with a const char* array
- add ternary operator for armOrMips condition
- use r_file_path for finding path to compilers
- new file created in libr/egg which contains all the C file parser
- modifications of 2 files to match the change :
  - libr/egg/r_egg.h
  - libr/egg/Makefile
- the function r_str_stripLine is simplier (the mallocs wasn't needed)
- the function r_str_stripLine is moved to libr/util/str.c
- libr/include/r_util/r_str.h is changed accordingly
This commit is contained in:
Gromimousse 2018-03-15 11:48:21 +01:00 committed by radare
parent 362184b722
commit a5c9fc7855
7 changed files with 418 additions and 6 deletions

View File

@ -11,7 +11,7 @@
#include <string.h>
static int usage (int v) {
static int usage(int v) {
printf ("Usage: ragg2 [-FOLsrxhvz] [-a arch] [-b bits] [-k os] [-o file] [-I path]\n"
" [-i sc] [-e enc] [-B hex] [-c k=v] [-C file] [-p pad] [-q off]\n"
" [-q off] [-dDw off:hex] file|f.asm|-\n");
@ -50,7 +50,8 @@ static int usage (int v) {
return 1;
}
static void list (REgg *egg) {
static void list(REgg *egg) {
RListIter *iter;
REggPlugin *p;
printf ("shellcodes:\n");
@ -67,7 +68,7 @@ static void list (REgg *egg) {
}
}
static int create (const char *format, const char *arch, int bits, const ut8 *code, int codelen) {
static int create(const char *format, const char *arch, int bits, const ut8 *code, int codelen) {
RBin *bin = r_bin_new ();
RBuffer *b;
if (!r_bin_use_arch (bin, arch, bits, format)) {
@ -86,7 +87,7 @@ static int create (const char *format, const char *arch, int bits, const ut8 *co
return 0;
}
static int openfile (const char *f, int x) {
static int openfile(const char *f, int x) {
int fd = open (f, O_RDWR | O_CREAT, 0644);
if (fd == -1) {
fd = open (f, O_RDWR);
@ -156,7 +157,7 @@ int main(int argc, char **argv) {
case 'C':
contents = optarg;
break;
case 'w':
case 'w':
{
char *arg = strdup (optarg);
char *p = strchr (arg, ':');
@ -191,7 +192,7 @@ int main(int argc, char **argv) {
ut64 n = r_num_math (NULL, optarg);
r_egg_patch (egg, -1, (const ut8*)&n, 8);
append = 1;
}
}
break;
case 'd':
{
@ -340,6 +341,28 @@ int main(int argc, char **argv) {
}
r_egg_load (egg, buf, 0);
}
} else if (strstr (file, ".c")) {
char *fileSanitized = strdup (file);
r_str_sanitize (fileSanitized);
char *textFile = r_egg_Cfile_parser (fileSanitized, arch, os, bits);
if (!textFile) {
eprintf ("Failure while parsing '%s'\n", fileSanitized);
goto fail;
}
int l;
char *buf = r_file_slurp (textFile, &l);
if (buf && l > 0) {
r_egg_raw (egg, (const ut8*)buf, l);
} else {
eprintf ("Error loading '%s'\n", textFile);
}
r_file_rm (textFile);
free (fileSanitized);
free (textFile);
free (buf);
} else {
if (strstr (file, ".s") || strstr (file, ".asm")) {
fmt = 'a';

View File

@ -9,6 +9,7 @@ OBJS+=emit_x86.o
OBJS+=emit_arm.o
OBJS+=emit_x64.o
OBJS+=emit_trace.o
OBJS+=egg_Cfile.o
all: ${LIBSO} ${LIBAR}

344
libr/egg/egg_Cfile.c Normal file
View File

@ -0,0 +1,344 @@
/* radare - LGPL - Copyright 2011-2018 - pancake */
#include <r_egg.h>
// compilation environment
struct cEnv_t {
char *SFLIBPATH;
char *CC;
const char *OBJCOPY;
char *CFLAGS;
char *LDFLAGS;
const char *JMP;
const char *FMT;
char *SHDR;
char *TRIPLET;
const char *TEXT;
};
static char* r_egg_Cfile_getCompiler(void) {
size_t i;
const char *compilers[] = {"llvm-gcc", "clang", "gcc"};
char *output = r_sys_getenv ("CC");
if (output) {
return output;
}
for (i = 0; i < 3; i++) {
output = r_file_path (compilers[i]);
if (strcmp (output, compilers[i])) {
free (output);
return strdup (compilers[i]);
}
free (output);
}
eprintf ("Couldn't find a compiler ! Please, set CC.\n");
return NULL;
}
static inline bool r_egg_Cfile_armOrMips(const char *arch) {
return (!strcmp (arch, "arm") || !strcmp (arch, "arm64") || !strcmp (arch, "aarch64")
|| !strcmp (arch, "thumb") || !strcmp (arch, "arm32") || !strcmp (arch, "mips")
|| !strcmp (arch, "mips32") || !strcmp (arch, "mips64"));
}
static void r_egg_Cfile_free_cEnv(struct cEnv_t *cEnv) {
if (cEnv) {
free (cEnv->SFLIBPATH);
free (cEnv->CC);
free (cEnv->CFLAGS);
free (cEnv->LDFLAGS);
free (cEnv->SHDR);
free (cEnv->TRIPLET);
}
free (cEnv);
}
static inline bool r_egg_Cfile_check_cEnv(struct cEnv_t *cEnv) {
return (!cEnv->SFLIBPATH || !cEnv->CC || !cEnv->CFLAGS || !cEnv->LDFLAGS
|| !cEnv->SHDR || !cEnv->TRIPLET);
}
static struct cEnv_t* r_egg_Cfile_set_cEnv(const char *arch, const char *os, int bits) {
struct cEnv_t *cEnv = calloc (1, sizeof (struct cEnv_t));
bool use_clang;
char *buffer = NULL;
char *output = NULL;
if (!cEnv) {
return NULL;
}
if (!(cEnv->CC = r_egg_Cfile_getCompiler())) {
goto fail;
}
cEnv->SFLIBPATH = r_sys_getenv ("SFLIBPATH");
if (!cEnv->SFLIBPATH) {
output = r_sys_cmd_strf ("r2 -hh | grep INCDIR | awk '{print $2}'");
if (!output || (output[0] == '\0')) {
eprintf ("Cannot find SFLIBPATH env var.\n"
"Please define it, or fix r2 installation.\n");
goto fail;
}
output[strlen (output) - 1] = '\0'; // strip the ending '\n'
if (!(cEnv->SFLIBPATH = r_str_newf ("%s/sflib", output))) {
goto fail;
}
}
cEnv->JMP = r_egg_Cfile_armOrMips (arch) ? "b" : "jmp";
if (!strcmp (os, "darwin")) {
cEnv->OBJCOPY = "gobjcopy";
cEnv->FMT = "mach0";
if (!strcmp (arch, "x86")) {
if (bits == 32) {
cEnv->CFLAGS = strdup ("-arch i386");
cEnv->LDFLAGS = strdup ("-arch i386 -shared -c");
} else {
cEnv->CFLAGS = strdup ("-arch x86_64");
cEnv->LDFLAGS = strdup ("-arch x86_64 -shared -c");
}
} else {
cEnv->LDFLAGS = strdup ("-shared -c");
}
cEnv->SHDR = r_str_newf ("\n.text\n%s _main\n", cEnv->JMP);
} else {
cEnv->OBJCOPY = "objcopy";
cEnv->FMT = "elf";
cEnv->SHDR = r_str_newf ("\n.section .text\n.globl main\n"
"// .type main, @function\n%s main\n", cEnv->JMP);
if (!strcmp (arch, "x86")) {
if (bits == 32) {
cEnv->CFLAGS = strdup ("-fPIC -fPIE -pie -fpic -m32");
cEnv->LDFLAGS = strdup ("-fPIC -fPIE -pie -fpic -m32");
} else {
cEnv->CFLAGS = strdup ("-fPIC -fPIE -pie -fpic -m64");
cEnv->LDFLAGS = strdup ("-fPIC -fPIE -pie -fpic -m64");
}
} else {
cEnv->CFLAGS = strdup ("-fPIC -fPIE -pie -fpic -nostartfiles");
cEnv->LDFLAGS = strdup ("-fPIC -fPIE -pie -fpic -nostartfiles");
}
}
cEnv->TRIPLET = r_str_newf ("%s-%s-%d", os, arch, bits);
if (!strcmp (os, "windows")) {
cEnv->TEXT = ".text";
cEnv->FMT = "pe";
} else if (!strcmp (os, "darwin")) {
cEnv->TEXT = "0.__TEXT.__text";
} else {
cEnv->TEXT = ".text";
}
use_clang = false;
if (!strcmp (cEnv->TRIPLET, "darwin-arm-64")) {
free (cEnv->CC);
cEnv->CC = strdup ("xcrun --sdk iphoneos gcc -arch arm64 -miphoneos-version-min=0.0");
use_clang = true;
cEnv->TEXT = "0.__TEXT.__text";
} else if (!strcmp (cEnv->TRIPLET, "darwin-arm-32")) {
free (cEnv->CC);
cEnv->CC = strdup ("xcrun --sdk iphoneos gcc -arch armv7 -miphoneos-version-min=0.0");
use_clang = true;
cEnv->TEXT = "0.__TEXT.__text";
}
buffer = r_str_newf ("%s -nostdinc -include '%s'/'%s'/sflib.h",
cEnv->CFLAGS, cEnv->SFLIBPATH, cEnv->TRIPLET);
if (!buffer) {
goto fail;
}
free (cEnv->CFLAGS);
cEnv->CFLAGS = strdup (buffer);
if (use_clang) {
free (buffer);
buffer = r_str_newf ("%s -fomit-frame-pointer"
" -fno-zero-initialized-in-bss", cEnv->CFLAGS);
if (!buffer) {
goto fail;
}
free (cEnv->CFLAGS);
cEnv->CFLAGS = strdup (buffer);
} else {
free (buffer);
buffer = r_str_newf ("%s -z execstack -fomit-frame-pointer"
" -finline-functions -fno-zero-initialized-in-bss", cEnv->CFLAGS);
if (!buffer) {
goto fail;
}
free (cEnv->CFLAGS);
cEnv->CFLAGS = strdup (buffer);
}
free (buffer);
buffer = r_str_newf ("%s -nostdlib", cEnv->LDFLAGS);
if (!buffer) {
goto fail;
}
free (cEnv->LDFLAGS);
cEnv->LDFLAGS = strdup (buffer);
if (r_egg_Cfile_check_cEnv (cEnv)) {
eprintf ("Error with cEnv allocation!\n");
goto fail;
}
free (buffer);
free (output);
return cEnv;
fail:
free (buffer);
free (output);
r_egg_Cfile_free_cEnv (cEnv);
return NULL;
}
static bool r_egg_Cfile_parseCompiled(const char *file) {
char *fileExt = r_str_newf ("%s.tmp", file);
char *buffer = r_file_slurp (fileExt, NULL);
buffer = r_str_replace (buffer, "rdata", "text", false);
buffer = r_str_replace (buffer, "rodata", "text", false);
buffer = r_str_replace (buffer, "get_pc_thunk.bx", "__getesp__", true);
const char *words[] = {".cstring", "size", "___main", "section", "__alloca", "zero", "cfi"};
size_t i;
for (i = 0; i < 7; i++) {
r_str_stripLine (buffer, words[i]);
}
free (fileExt);
fileExt = r_str_newf ("%s.s", file);
if (!r_file_dump (fileExt, (const ut8*) buffer, strlen (buffer), true)) {
eprintf ("Error while opening %s.s\n", file);
goto fail;
}
free (buffer);
free (fileExt);
return true;
fail:
free (buffer);
free (fileExt);
return false;
}
R_API char* r_egg_Cfile_parser(const char *file, const char *arch, const char *os, int bits) {
char *output = NULL;
char *fileExt = NULL; // "file" with extension (.s, .text, ...)
struct cEnv_t *cEnv = r_egg_Cfile_set_cEnv (arch, os, bits);
if (!cEnv) {
goto fail;
}
r_str_sanitize (cEnv->CC);
//printf ("==> Compile\n");
printf ("'%s' %s -o '%s.tmp' -S -Os '%s'\n", cEnv->CC, cEnv->CFLAGS, file, file);
output = r_sys_cmd_strf ("('%s' %s -o '%s.tmp' -S -Os '%s') 2>&1",
cEnv->CC, cEnv->CFLAGS, file, file);
if (output == NULL) {
eprintf ("Compilation failed!\n");
goto fail;
}
printf ("%s", output);
if (!(fileExt = r_str_newf ("%s.s", file))) {
goto fail;
}
if (!r_file_dump (fileExt, (const ut8*) cEnv->SHDR, strlen (cEnv->SHDR), false)) {
eprintf ("Error while opening %s.s\n", file);
goto fail;
}
if (!r_egg_Cfile_parseCompiled (file)) {
goto fail;
}
//printf ("==> Assemble\n");
printf ("'%s' %s -Os -o '%s.o' '%s.s'\n", cEnv->CC, cEnv->LDFLAGS, file, file);
free (output);
output = r_sys_cmd_strf ("'%s' %s -Os -o '%s.o' '%s.s'",
cEnv->CC, cEnv->LDFLAGS, file, file);
if (!output) {
eprintf ("Assembly failed!\n");
goto fail;
}
printf ("%s", output);
//printf ("==> Link\n");
printf ("rabin2 -o '%s.text' -O d/S/'%s' '%s.o'\n", file, cEnv->TEXT, file);
free (output);
output = r_sys_cmd_strf ("rabin2 -o '%s.text' -O d/S/'%s' '%s'.o",
file, cEnv->TEXT, file);
if (!output) {
eprintf ("Linkage failed!\n");
goto fail;
}
free (fileExt);
if (!(fileExt = r_str_newf ("%s.o", file))) {
goto fail;
}
if (!r_file_exists (fileExt)) {
eprintf ("Cannot find %s.o\n", file);
goto fail;
}
free (fileExt);
if (!(fileExt = r_str_newf ("%s.text", file))) {
goto fail;
}
if (r_file_size (fileExt) == 0) {
printf ("FALLBACK: Using objcopy instead of rabin2");
free (output);
output = r_sys_cmd_strf ("'%s' -j .text -O binary '%s.o' '%s.text'",
cEnv->OBJCOPY, file, file);
if (!output) {
eprintf ("objcopy failed!\n");
goto fail;
}
}
size_t i;
const char *extArray[] = {"bin", "tmp", "s", "o"};
for (i = 0; i < 4; i++) {
free (fileExt);
if (!(fileExt = r_str_newf ("%s.%s", file, extArray[i]))) {
goto fail;
}
r_file_rm (fileExt);
}
free (fileExt);
if ((fileExt = r_str_newf ("%s.text", file)) == NULL) {
goto fail;
}
free (output);
r_egg_Cfile_free_cEnv (cEnv);
return fileExt;
fail:
free (fileExt);
free (output);
r_egg_Cfile_free_cEnv (cEnv);
return NULL;
}

View File

@ -1,5 +1,6 @@
files = [
'egg.c',
'egg_Cfile.c',
'egg_lang.c',
'emit_arm.c',
'emit_trace.c',

View File

@ -217,6 +217,9 @@ R_API int r_egg_run(REgg *egg);
R_API int r_egg_patch(REgg *egg, int off, const ut8 *b, int l);
R_API void r_egg_finalize(REgg *egg);
/* r_egg_Cfile.c */
R_API char* r_egg_Cfile_parser(const char *file, const char *arch, const char *os, int bits);
/* lang.c */
R_API char *r_egg_mkvar(REgg *egg, char *out, const char *_str, int delta);
R_API int r_egg_lang_parsechar(REgg *egg, char c);

View File

@ -158,6 +158,7 @@ R_API const char *r_str_last (const char *in, const char *ch);
R_API char* r_str_highlight(char *str, const char *word, const char *color);
R_API char *r_qrcode_gen(const ut8 *text, int len, bool utf8, bool inverted);
R_API char *r_str_from_ut64(ut64 val);
R_API void r_str_stripLine(char *str, const char *key);
#ifdef __cplusplus
}
#endif

View File

@ -2936,3 +2936,42 @@ R_API int r_snprintf(char *string, int len, const char *fmt, ...) {
va_end (ap);
return ret;
}
// Strips all the lines in str that contain key
R_API void r_str_stripLine(char *str, const char *key)
{
size_t i, j, klen, slen, off;
const char *ptr;
if (!str || !key) {
return;
}
klen = strlen (key);
slen = strlen (str);
for (i = 0; i < slen; ) {
ptr = (char*) r_mem_mem ((ut8*) str + i, slen - i, (ut8*) "\n", 1);
if (!ptr) {
ptr = (char*) r_mem_mem ((ut8*) str + i, slen - i, (ut8*) key, klen);
if (ptr) {
str[i] = '\0';
break;
}
break;
}
off = (size_t) (ptr - (str + i)) + 1;
ptr = (char*) r_mem_mem ((ut8*) str + i, off, (ut8*) key, klen);
if (ptr) {
for (j = i; j < slen - off + 1; j++) {
str[j] = str[j + off];
}
slen -= off;
} else {
i += off;
}
}
return;
}