Esempio n. 1
0
void emu_track_and_source_free(struct emu_track_and_source *et)
{
	if (et->static_instr_table != NULL)
		emu_hashtable_free(et->static_instr_table);

	if (et->static_instr_graph != NULL)
		emu_graph_free(et->static_instr_graph);

	if (et->run_instr_table != NULL)
		emu_hashtable_free(et->run_instr_table);

	if (et->run_instr_graph != NULL)
		emu_graph_free(et->run_instr_graph);

	free(et);

}
Esempio n. 2
0
void emu_env_linux_free(struct emu_env_linux *eel)
{
	emu_hashtable_free(eel->syscall_hooks_by_name);
	free(eel->syscall_hookx);
	free(eel->hooks);
//	emu_profile_free(eel->profile);
	free(eel);
}
Esempio n. 3
0
File: dot.c Progetto: tpltnt/libemu
int graph_draw(struct emu_graph *graph)
{
	struct emu_vertex *ev;
	struct instr_vertex *iv;

	FILE *f = fopen(opts.graphfile,"w+");

	struct emu_graph *sgraph = emu_graph_new();
	struct emu_hashtable *ht = emu_hashtable_new(2047, emu_hashtable_ptr_hash, emu_hashtable_ptr_cmp);

	struct emu_vertex *nev;
	struct instr_vertex *niv=NULL;

	printf("copying vertexes\n");
	for ( ev = emu_vertexes_first(graph->vertexes); !emu_vertexes_attail(ev); ev = emu_vertexes_next(ev) )
	{
		iv = (struct instr_vertex *)ev->data;

		nev = emu_vertex_new();
		emu_graph_vertex_add(sgraph, nev);

		niv = instr_vertex_copy(iv); 
		nev->data = niv;

		emu_hashtable_insert(ht, (void *)iv, nev);
		ev->color = white;
	}

	printf("optimizing graph\n");
	for ( ev = emu_vertexes_first(graph->vertexes); !emu_vertexes_attail(ev); ev = emu_vertexes_next(ev) )
	{
		// ignore known 
		if ( ev->color == black )
			continue;


		printf("vertex %p\n", (void *)ev);

		// find the first in a chain
		iv = (struct instr_vertex *)ev->data;
		while ( emu_edges_length(ev->backedges) == 1 && emu_edges_length(ev->edges) <= 1 && ev->color == white && iv->dll == NULL && iv->syscall == NULL )
		{
			ev->color = grey;

			struct emu_vertex *xev = emu_edges_first(ev->backedges)->destination;
			iv = (struct instr_vertex *)xev->data;
			if ( emu_edges_length(xev->backedges) > 1 || emu_edges_length(xev->edges) > 1 || iv->dll != NULL || iv->syscall != NULL )
				break;

			ev = xev;
			printf(" -> vertex %p\n",(void *)ev);
		}


		iv = (struct instr_vertex *)ev->data;

		// create the new vertex 
		nev = (struct emu_vertex *)emu_hashtable_search(ht, (void *)iv)->value;
		niv = (struct instr_vertex *)nev->data;

		iv = (struct instr_vertex *)ev->data;

		printf("going forwards from %p\n", (void *)ev);
		while ( emu_edges_length(ev->edges) == 1 && emu_edges_length(ev->backedges) <= 1 && ev->color != black && iv->dll == NULL && iv->syscall == NULL )
		{
			ev->color = black;
			struct emu_vertex *xev = emu_edges_first(ev->edges)->destination;
			iv = (struct instr_vertex *)xev->data;

			if ( emu_edges_length(xev->backedges) > 1 || emu_edges_length(xev->edges) > 1 ||
				 iv->dll != NULL || iv->syscall != NULL )
				break;

			ev = xev;

			iv = (struct instr_vertex *)ev->data;
			emu_string_append_char(niv->instr_string, emu_string_char(iv->instr_string));
			printf(" -> vertex %p\n",(void *)ev);
		}

		ev->color = black;

		printf("copying edges for %p\n",(void *)ev);
		struct emu_edge *ee;
		for ( ee = emu_edges_first(ev->edges); !emu_edges_attail(ee); ee = emu_edges_next(ee) )
		{
			struct instr_vertex *ivto = emu_vertex_data_get(ee->destination);
			struct emu_hashtable_item *ehi = emu_hashtable_search(ht, (void *)ivto);
			struct emu_vertex *to = (struct emu_vertex *)ehi->value;
			if ( 1 )// nev != to )//&& to->color != black )
			{
				struct emu_edge *nee = emu_vertex_edge_add(nev, to);
				nee->count = ee->count;
				nee->data = ee->data;
				printf(" -> %p\n", (void *)to);
			}
		}

	}

	graph->vertex_destructor = instr_vertex_destructor;
//	emu_graph_free(graph);
	graph = sgraph;

	emu_hashtable_free(ht);
	fprintf(f, "digraph G {\n\t//rankdir=LR\n\tnode [fontname=Courier, labeljust=r];\n");

#if 0
	int numdlls=0;
	while ( env->loaded_dlls[numdlls] != NULL )
	{
		int has_dll = 0;
		struct emu_string *fs = emu_string_new();
		emu_string_append_format(fs, "\t subgraph cluster%i {\n\t\t node [shape=box, style=filled, color=\".7 .3 1.0\"];\n\t\tstyle=filled;\n\t\tcolor=lightgrey;\n\t\tlabel=\"%s\"\n\t\t", numdlls, env->loaded_dlls[numdlls]->dllname);
		for ( ev = emu_vertexes_first(graph->vertexes); !emu_vertexes_attail(ev); ev = emu_vertexes_next(ev) )
		{
			struct instr_vertex *iv = emu_vertex_data_get(ev);
			if ( iv->dll == env->loaded_dlls[numdlls] )
			{
				emu_string_append_format(fs, "\t\%i [label = \"%s\"];\n", iv->eip, emu_string_char(iv->instr_string));

				has_dll = 1;
			}

		}

		emu_string_append_char(fs, "\t}\n");

		fprintf(f, "%s", emu_string_char(fs));
		numdlls++;
	}
Esempio n. 4
0
int32_t emu_shellcode_test(struct emu *e, uint8_t *data, uint16_t size)
{
	logPF(e);
/*

	
	bool found_good_candidate_after_getpc = false;

	uint32_t best_eip=0;
*/
	// This check avoids a floating point exception further down the line
	if(size < 2)
	{
		return -1;
	}

	uint32_t offset;
	struct emu_list_root *el;
	el = emu_list_create();


	for ( offset=0; offset<size ; offset++ )
	{
		if ( emu_getpc_check(e, (uint8_t *)data, size, offset) != 0 )
		{
			logDebug(e, "possible getpc at offset %i (%08x)\n", offset, offset);
			struct emu_list_item *eli = emu_list_item_create();
			eli->uint32 = offset;
			emu_list_insert_last(el, eli);
		}
	}

	if ( emu_list_length(el) == 0 )
	{
		emu_list_destroy(el);
		return -1;
	}

	{
		struct emu_cpu *cpu = emu_cpu_get(e);
		struct emu_memory *mem = emu_memory_get(e);

		/* write the code to the offset */
		emu_memory_write_block(mem, STATIC_OFFSET, data, size);

		/* set the registers to the initial values */
		int reg;
		for ( reg=0;reg<8;reg++ )
			emu_cpu_reg32_set(cpu,reg ,0x0);

		emu_cpu_reg32_set(cpu, esp, 0x00120000);
		emu_cpu_eip_set(cpu, 0);

		/* set the flags */
		emu_cpu_eflags_set(cpu,0x0);
	}

	struct emu_track_and_source *etas = emu_track_and_source_new();

	logDebug(e, "creating static callgraph\n");
	/* create the static analysis graph 
	set memory read only so instructions can't change it*/
	emu_memory_mode_ro(emu_memory_get(e));
	emu_source_instruction_graph_create(e, etas, STATIC_OFFSET, size);
	emu_memory_mode_rw(emu_memory_get(e));

	struct emu_hashtable *eh = emu_hashtable_new(size+4/4, emu_hashtable_ptr_hash, emu_hashtable_ptr_cmp);
	struct emu_list_item *eli;
//	struct emu_env_w32 *env = emu_env_w32_new(e);
	

	struct emu_list_root *results = emu_list_create();

	for ( eli = emu_list_first(el); !emu_list_attail(eli); eli = emu_list_next(eli) )
	{
		logDebug(e, "testing offset %i %08x\n", eli->uint32, eli->uint32);
		emu_shellcode_run_and_track(e, data, size, eli->uint32, 256, etas, eh,
									results, false);
	}

	

	/* for all positions we got, take the best, maybe take memory access into account later */
	emu_list_qsort(results, tested_positions_cmp);
	if ( ((struct emu_stats *)emu_list_first(results)->data)->cpu.steps != 256 )
	{
		emu_hashtable_free(eh);
		eh = emu_hashtable_new(size+4/4, emu_hashtable_ptr_hash, emu_hashtable_ptr_cmp);
		logDebug(e, "brute force!\n");
		struct emu_list_root *new_results = emu_list_create();
		for ( eli = emu_list_first(results); !emu_list_attail(eli); eli = emu_list_next(eli) )
		{
			struct emu_stats *es = (struct emu_stats *)eli->data;
			logDebug(e, "brute at offset 0x%08x \n",es->eip - STATIC_OFFSET);
			emu_shellcode_run_and_track(e, data, size, es->eip - STATIC_OFFSET, 256, etas, eh,
										new_results, true);
			
		}

		emu_list_concat(results, new_results);
		emu_list_destroy(new_results);
		emu_list_qsort(results, tested_positions_cmp);

		/* uniq */
		for ( eli = emu_list_first(results); !emu_list_attail(eli); eli = emu_list_next(eli) )
		{
			struct emu_list_item *next = emu_list_next(eli);
			if (!emu_list_attail(next) &&
            	((struct emu_stats *)eli->data)->eip == ((struct emu_stats *)next->data)->eip )
			{
				emu_stats_free(next->data);
				emu_list_remove(next);
				free(next);
			}
		}
	}

	

	emu_hashtable_free(eh);
	emu_list_destroy(el);
//	emu_env_w32_free(env);
	emu_track_and_source_free(etas);


	{
		struct emu_list_item *eli;
		for ( eli = emu_list_first(results); !emu_list_attail(eli); eli = emu_list_next(eli) )
		{
			struct emu_stats *es = (struct emu_stats *)eli->data;
			logDebug(e, "b offset 0x%08x steps %i\n",es->eip, es->cpu.steps);
		}
	}

	eli = emu_list_first(results);
	struct emu_stats *es = (struct emu_stats *)eli->data;

	if ( es->cpu.steps > 100 )
	{
		offset = es->eip;
	}
	else
	{
		offset = -1;
	}

	for (eli = emu_list_first(results); !emu_list_attail(eli); eli = emu_list_next(eli))
	{
		emu_stats_free((struct emu_stats *)eli->data);
	}
	emu_list_destroy(results);

	return offset - STATIC_OFFSET;
}