| 
									
										
										
										
											2012-10-27 23:18:32 +02:00
										 |  |  | #include "symbol.h"
 | 
					
						
							|  |  |  | #include "dso.h"
 | 
					
						
							| 
									
										
										
										
											2012-11-09 11:32:52 -03:00
										 |  |  | #include "machine.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-27 23:18:32 +02:00
										 |  |  | #include "util.h"
 | 
					
						
							|  |  |  | #include "debug.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char dso__symtab_origin(const struct dso *dso) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static const char origin[] = { | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__KALLSYMS]		= 'k', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__VMLINUX]		= 'v', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__JAVA_JIT]		= 'j', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__DEBUGLINK]		= 'l', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__BUILD_ID_CACHE]	= 'B', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__FEDORA_DEBUGINFO]	= 'f', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__UBUNTU_DEBUGINFO]	= 'u', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__BUILDID_DEBUGINFO]	= 'b', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__SYSTEM_PATH_DSO]	= 'd', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__SYSTEM_PATH_KMODULE]	= 'K', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__GUEST_KALLSYMS]	= 'g', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__GUEST_KMODULE]	= 'G', | 
					
						
							|  |  |  | 		[DSO_BINARY_TYPE__GUEST_VMLINUX]	= 'V', | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dso == NULL || dso->symtab_type == DSO_BINARY_TYPE__NOT_FOUND) | 
					
						
							|  |  |  | 		return '!'; | 
					
						
							|  |  |  | 	return origin[dso->symtab_type]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dso__binary_type_file(struct dso *dso, enum dso_binary_type type, | 
					
						
							|  |  |  | 			  char *root_dir, char *file, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char build_id_hex[BUILD_ID_SIZE * 2 + 1]; | 
					
						
							|  |  |  | 	int ret = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (type) { | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__DEBUGLINK: { | 
					
						
							|  |  |  | 		char *debuglink; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		strncpy(file, dso->long_name, size); | 
					
						
							|  |  |  | 		debuglink = file + dso->long_name_len; | 
					
						
							|  |  |  | 		while (debuglink != file && *debuglink != '/') | 
					
						
							|  |  |  | 			debuglink--; | 
					
						
							|  |  |  | 		if (*debuglink == '/') | 
					
						
							|  |  |  | 			debuglink++; | 
					
						
							|  |  |  | 		filename__read_debuglink(dso->long_name, debuglink, | 
					
						
							|  |  |  | 					 size - (debuglink - file)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__BUILD_ID_CACHE: | 
					
						
							|  |  |  | 		/* skip the locally configured cache if a symfs is given */ | 
					
						
							|  |  |  | 		if (symbol_conf.symfs[0] || | 
					
						
							|  |  |  | 		    (dso__build_id_filename(dso, file, size) == NULL)) | 
					
						
							|  |  |  | 			ret = -1; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__FEDORA_DEBUGINFO: | 
					
						
							|  |  |  | 		snprintf(file, size, "%s/usr/lib/debug%s.debug", | 
					
						
							|  |  |  | 			 symbol_conf.symfs, dso->long_name); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__UBUNTU_DEBUGINFO: | 
					
						
							|  |  |  | 		snprintf(file, size, "%s/usr/lib/debug%s", | 
					
						
							|  |  |  | 			 symbol_conf.symfs, dso->long_name); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__BUILDID_DEBUGINFO: | 
					
						
							|  |  |  | 		if (!dso->has_build_id) { | 
					
						
							|  |  |  | 			ret = -1; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		build_id__sprintf(dso->build_id, | 
					
						
							|  |  |  | 				  sizeof(dso->build_id), | 
					
						
							|  |  |  | 				  build_id_hex); | 
					
						
							|  |  |  | 		snprintf(file, size, | 
					
						
							|  |  |  | 			 "%s/usr/lib/debug/.build-id/%.2s/%s.debug", | 
					
						
							|  |  |  | 			 symbol_conf.symfs, build_id_hex, build_id_hex + 2); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__SYSTEM_PATH_DSO: | 
					
						
							|  |  |  | 		snprintf(file, size, "%s%s", | 
					
						
							|  |  |  | 			 symbol_conf.symfs, dso->long_name); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__GUEST_KMODULE: | 
					
						
							|  |  |  | 		snprintf(file, size, "%s%s%s", symbol_conf.symfs, | 
					
						
							|  |  |  | 			 root_dir, dso->long_name); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__SYSTEM_PATH_KMODULE: | 
					
						
							|  |  |  | 		snprintf(file, size, "%s%s", symbol_conf.symfs, | 
					
						
							|  |  |  | 			 dso->long_name); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__KALLSYMS: | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__VMLINUX: | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__GUEST_KALLSYMS: | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__GUEST_VMLINUX: | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__JAVA_JIT: | 
					
						
							|  |  |  | 	case DSO_BINARY_TYPE__NOT_FOUND: | 
					
						
							|  |  |  | 		ret = -1; | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int open_dso(struct dso *dso, struct machine *machine) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *root_dir = (char *) ""; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | 	int fd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	name = malloc(PATH_MAX); | 
					
						
							|  |  |  | 	if (!name) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (machine) | 
					
						
							|  |  |  | 		root_dir = machine->root_dir; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dso__binary_type_file(dso, dso->data_type, | 
					
						
							|  |  |  | 				  root_dir, name, PATH_MAX)) { | 
					
						
							|  |  |  | 		free(name); | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fd = open(name, O_RDONLY); | 
					
						
							|  |  |  | 	free(name); | 
					
						
							|  |  |  | 	return fd; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dso__data_fd(struct dso *dso, struct machine *machine) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static enum dso_binary_type binary_type_data[] = { | 
					
						
							|  |  |  | 		DSO_BINARY_TYPE__BUILD_ID_CACHE, | 
					
						
							|  |  |  | 		DSO_BINARY_TYPE__SYSTEM_PATH_DSO, | 
					
						
							|  |  |  | 		DSO_BINARY_TYPE__NOT_FOUND, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	int i = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dso->data_type != DSO_BINARY_TYPE__NOT_FOUND) | 
					
						
							|  |  |  | 		return open_dso(dso, machine); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		int fd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		dso->data_type = binary_type_data[i++]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		fd = open_dso(dso, machine); | 
					
						
							|  |  |  | 		if (fd >= 0) | 
					
						
							|  |  |  | 			return fd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} while (dso->data_type != DSO_BINARY_TYPE__NOT_FOUND); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return -EINVAL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | dso_cache__free(struct rb_root *root) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *next = rb_first(root); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (next) { | 
					
						
							|  |  |  | 		struct dso_cache *cache; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cache = rb_entry(next, struct dso_cache, rb_node); | 
					
						
							|  |  |  | 		next = rb_next(&cache->rb_node); | 
					
						
							|  |  |  | 		rb_erase(&cache->rb_node, root); | 
					
						
							|  |  |  | 		free(cache); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct dso_cache* | 
					
						
							|  |  |  | dso_cache__find(struct rb_root *root, u64 offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node **p = &root->rb_node; | 
					
						
							|  |  |  | 	struct rb_node *parent = NULL; | 
					
						
							|  |  |  | 	struct dso_cache *cache; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (*p != NULL) { | 
					
						
							|  |  |  | 		u64 end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		parent = *p; | 
					
						
							|  |  |  | 		cache = rb_entry(parent, struct dso_cache, rb_node); | 
					
						
							|  |  |  | 		end = cache->offset + DSO__DATA_CACHE_SIZE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (offset < cache->offset) | 
					
						
							|  |  |  | 			p = &(*p)->rb_left; | 
					
						
							|  |  |  | 		else if (offset >= end) | 
					
						
							|  |  |  | 			p = &(*p)->rb_right; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			return cache; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | dso_cache__insert(struct rb_root *root, struct dso_cache *new) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node **p = &root->rb_node; | 
					
						
							|  |  |  | 	struct rb_node *parent = NULL; | 
					
						
							|  |  |  | 	struct dso_cache *cache; | 
					
						
							|  |  |  | 	u64 offset = new->offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (*p != NULL) { | 
					
						
							|  |  |  | 		u64 end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		parent = *p; | 
					
						
							|  |  |  | 		cache = rb_entry(parent, struct dso_cache, rb_node); | 
					
						
							|  |  |  | 		end = cache->offset + DSO__DATA_CACHE_SIZE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (offset < cache->offset) | 
					
						
							|  |  |  | 			p = &(*p)->rb_left; | 
					
						
							|  |  |  | 		else if (offset >= end) | 
					
						
							|  |  |  | 			p = &(*p)->rb_right; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rb_link_node(&new->rb_node, parent, p); | 
					
						
							|  |  |  | 	rb_insert_color(&new->rb_node, root); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t | 
					
						
							|  |  |  | dso_cache__memcpy(struct dso_cache *cache, u64 offset, | 
					
						
							|  |  |  | 		  u8 *data, u64 size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	u64 cache_offset = offset - cache->offset; | 
					
						
							|  |  |  | 	u64 cache_size   = min(cache->size - cache_offset, size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	memcpy(data, cache->data + cache_offset, cache_size); | 
					
						
							|  |  |  | 	return cache_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t | 
					
						
							|  |  |  | dso_cache__read(struct dso *dso, struct machine *machine, | 
					
						
							|  |  |  | 		 u64 offset, u8 *data, ssize_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dso_cache *cache; | 
					
						
							|  |  |  | 	ssize_t ret; | 
					
						
							|  |  |  | 	int fd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fd = dso__data_fd(dso, machine); | 
					
						
							|  |  |  | 	if (fd < 0) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		u64 cache_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ret = -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cache = zalloc(sizeof(*cache) + DSO__DATA_CACHE_SIZE); | 
					
						
							|  |  |  | 		if (!cache) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cache_offset = offset & DSO__DATA_CACHE_MASK; | 
					
						
							|  |  |  | 		ret = -EINVAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (-1 == lseek(fd, cache_offset, SEEK_SET)) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ret = read(fd, cache->data, DSO__DATA_CACHE_SIZE); | 
					
						
							|  |  |  | 		if (ret <= 0) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cache->offset = cache_offset; | 
					
						
							|  |  |  | 		cache->size   = ret; | 
					
						
							|  |  |  | 		dso_cache__insert(&dso->cache, cache); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ret = dso_cache__memcpy(cache, offset, data, size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} while (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ret <= 0) | 
					
						
							|  |  |  | 		free(cache); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	close(fd); | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static ssize_t dso_cache_read(struct dso *dso, struct machine *machine, | 
					
						
							|  |  |  | 			      u64 offset, u8 *data, ssize_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dso_cache *cache; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	cache = dso_cache__find(&dso->cache, offset); | 
					
						
							|  |  |  | 	if (cache) | 
					
						
							|  |  |  | 		return dso_cache__memcpy(cache, offset, data, size); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return dso_cache__read(dso, machine, offset, data, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ssize_t dso__data_read_offset(struct dso *dso, struct machine *machine, | 
					
						
							|  |  |  | 			      u64 offset, u8 *data, ssize_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ssize_t r = 0; | 
					
						
							|  |  |  | 	u8 *p = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	do { | 
					
						
							|  |  |  | 		ssize_t ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ret = dso_cache_read(dso, machine, offset, p, size); | 
					
						
							|  |  |  | 		if (ret < 0) | 
					
						
							|  |  |  | 			return ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* Reached EOF, return what we have. */ | 
					
						
							|  |  |  | 		if (!ret) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BUG_ON(ret > size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		r      += ret; | 
					
						
							|  |  |  | 		p      += ret; | 
					
						
							|  |  |  | 		offset += ret; | 
					
						
							|  |  |  | 		size   -= ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} while (size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ssize_t dso__data_read_addr(struct dso *dso, struct map *map, | 
					
						
							|  |  |  | 			    struct machine *machine, u64 addr, | 
					
						
							|  |  |  | 			    u8 *data, ssize_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	u64 offset = map->map_ip(map, addr); | 
					
						
							|  |  |  | 	return dso__data_read_offset(dso, machine, offset, data, size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct map *dso__new_map(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct map *map = NULL; | 
					
						
							|  |  |  | 	struct dso *dso = dso__new(name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dso) | 
					
						
							|  |  |  | 		map = map__new2(0, dso, MAP__FUNCTION); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return map; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct dso *dso__kernel_findnew(struct machine *machine, const char *name, | 
					
						
							|  |  |  | 		    const char *short_name, int dso_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * The kernel dso could be created by build_id processing. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	struct dso *dso = __dsos__findnew(&machine->kernel_dsos, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * We need to run this in all cases, since during the build_id | 
					
						
							|  |  |  | 	 * processing we had no idea this was the kernel dso. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (dso != NULL) { | 
					
						
							|  |  |  | 		dso__set_short_name(dso, short_name); | 
					
						
							|  |  |  | 		dso->kernel = dso_type; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return dso; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dso__set_long_name(struct dso *dso, char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (name == NULL) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	dso->long_name = name; | 
					
						
							|  |  |  | 	dso->long_name_len = strlen(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dso__set_short_name(struct dso *dso, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (name == NULL) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	dso->short_name = name; | 
					
						
							|  |  |  | 	dso->short_name_len = strlen(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void dso__set_basename(struct dso *dso) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	dso__set_short_name(dso, basename(dso->long_name)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dso__name_len(const struct dso *dso) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!dso) | 
					
						
							|  |  |  | 		return strlen("[unknown]"); | 
					
						
							|  |  |  | 	if (verbose) | 
					
						
							|  |  |  | 		return dso->long_name_len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return dso->short_name_len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool dso__loaded(const struct dso *dso, enum map_type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return dso->loaded & (1 << type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool dso__sorted_by_name(const struct dso *dso, enum map_type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return dso->sorted_by_name & (1 << type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dso__set_sorted_by_name(struct dso *dso, enum map_type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	dso->sorted_by_name |= (1 << type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct dso *dso__new(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dso *dso = calloc(1, sizeof(*dso) + strlen(name) + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dso != NULL) { | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		strcpy(dso->name, name); | 
					
						
							|  |  |  | 		dso__set_long_name(dso, dso->name); | 
					
						
							|  |  |  | 		dso__set_short_name(dso, dso->name); | 
					
						
							|  |  |  | 		for (i = 0; i < MAP__NR_TYPES; ++i) | 
					
						
							|  |  |  | 			dso->symbols[i] = dso->symbol_names[i] = RB_ROOT; | 
					
						
							|  |  |  | 		dso->cache = RB_ROOT; | 
					
						
							|  |  |  | 		dso->symtab_type = DSO_BINARY_TYPE__NOT_FOUND; | 
					
						
							|  |  |  | 		dso->data_type   = DSO_BINARY_TYPE__NOT_FOUND; | 
					
						
							|  |  |  | 		dso->loaded = 0; | 
					
						
							|  |  |  | 		dso->sorted_by_name = 0; | 
					
						
							|  |  |  | 		dso->has_build_id = 0; | 
					
						
							|  |  |  | 		dso->kernel = DSO_TYPE_USER; | 
					
						
							|  |  |  | 		dso->needs_swap = DSO_SWAP__UNSET; | 
					
						
							|  |  |  | 		INIT_LIST_HEAD(&dso->node); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return dso; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dso__delete(struct dso *dso) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	for (i = 0; i < MAP__NR_TYPES; ++i) | 
					
						
							|  |  |  | 		symbols__delete(&dso->symbols[i]); | 
					
						
							|  |  |  | 	if (dso->sname_alloc) | 
					
						
							|  |  |  | 		free((char *)dso->short_name); | 
					
						
							|  |  |  | 	if (dso->lname_alloc) | 
					
						
							|  |  |  | 		free(dso->long_name); | 
					
						
							|  |  |  | 	dso_cache__free(&dso->cache); | 
					
						
							|  |  |  | 	free(dso); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dso__set_build_id(struct dso *dso, void *build_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	memcpy(dso->build_id, build_id, sizeof(dso->build_id)); | 
					
						
							|  |  |  | 	dso->has_build_id = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool dso__build_id_equal(const struct dso *dso, u8 *build_id) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return memcmp(dso->build_id, build_id, sizeof(dso->build_id)) == 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dso__read_running_kernel_build_id(struct dso *dso, struct machine *machine) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char path[PATH_MAX]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (machine__is_default_guest(machine)) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	sprintf(path, "%s/sys/kernel/notes", machine->root_dir); | 
					
						
							|  |  |  | 	if (sysfs__read_build_id(path, dso->build_id, | 
					
						
							|  |  |  | 				 sizeof(dso->build_id)) == 0) | 
					
						
							|  |  |  | 		dso->has_build_id = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dso__kernel_module_get_build_id(struct dso *dso, | 
					
						
							|  |  |  | 				    const char *root_dir) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char filename[PATH_MAX]; | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * kernel module short names are of the form "[module]" and | 
					
						
							|  |  |  | 	 * we need just "module" here. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	const char *name = dso->short_name + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	snprintf(filename, sizeof(filename), | 
					
						
							|  |  |  | 		 "%s/sys/module/%.*s/notes/.note.gnu.build-id", | 
					
						
							|  |  |  | 		 root_dir, (int)strlen(name) - 1, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (sysfs__read_build_id(filename, dso->build_id, | 
					
						
							|  |  |  | 				 sizeof(dso->build_id)) == 0) | 
					
						
							|  |  |  | 		dso->has_build_id = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool __dsos__read_build_ids(struct list_head *head, bool with_hits) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bool have_build_id = false; | 
					
						
							|  |  |  | 	struct dso *pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(pos, head, node) { | 
					
						
							|  |  |  | 		if (with_hits && !pos->hit) | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		if (pos->has_build_id) { | 
					
						
							|  |  |  | 			have_build_id = true; | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (filename__read_build_id(pos->long_name, pos->build_id, | 
					
						
							|  |  |  | 					    sizeof(pos->build_id)) > 0) { | 
					
						
							|  |  |  | 			have_build_id	  = true; | 
					
						
							|  |  |  | 			pos->has_build_id = true; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return have_build_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dsos__add(struct list_head *head, struct dso *dso) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	list_add_tail(&dso->node, head); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct dso *dsos__find(struct list_head *head, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dso *pos; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(pos, head, node) | 
					
						
							|  |  |  | 		if (strcmp(pos->long_name, name) == 0) | 
					
						
							|  |  |  | 			return pos; | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct dso *__dsos__findnew(struct list_head *head, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dso *dso = dsos__find(head, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!dso) { | 
					
						
							|  |  |  | 		dso = dso__new(name); | 
					
						
							|  |  |  | 		if (dso != NULL) { | 
					
						
							|  |  |  | 			dsos__add(head, dso); | 
					
						
							|  |  |  | 			dso__set_basename(dso); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return dso; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t __dsos__fprintf_buildid(struct list_head *head, FILE *fp, | 
					
						
							| 
									
										
										
										
											2012-12-07 09:53:58 -03:00
										 |  |  | 			       bool (skip)(struct dso *dso, int parm), int parm) | 
					
						
							| 
									
										
										
										
											2012-10-27 23:18:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct dso *pos; | 
					
						
							|  |  |  | 	size_t ret = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(pos, head, node) { | 
					
						
							| 
									
										
										
										
											2012-12-07 09:53:58 -03:00
										 |  |  | 		if (skip && skip(pos, parm)) | 
					
						
							| 
									
										
										
										
											2012-10-27 23:18:32 +02:00
										 |  |  | 			continue; | 
					
						
							|  |  |  | 		ret += dso__fprintf_buildid(pos, fp); | 
					
						
							|  |  |  | 		ret += fprintf(fp, " %s\n", pos->long_name); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t __dsos__fprintf(struct list_head *head, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct dso *pos; | 
					
						
							|  |  |  | 	size_t ret = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(pos, head, node) { | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		for (i = 0; i < MAP__NR_TYPES; ++i) | 
					
						
							|  |  |  | 			ret += dso__fprintf(pos, i, fp); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t dso__fprintf_buildid(struct dso *dso, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char sbuild_id[BUILD_ID_SIZE * 2 + 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	build_id__sprintf(dso->build_id, sizeof(dso->build_id), sbuild_id); | 
					
						
							|  |  |  | 	return fprintf(fp, "%s", sbuild_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t dso__fprintf(struct dso *dso, enum map_type type, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct rb_node *nd; | 
					
						
							|  |  |  | 	size_t ret = fprintf(fp, "dso: %s (", dso->short_name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dso->short_name != dso->long_name) | 
					
						
							|  |  |  | 		ret += fprintf(fp, "%s, ", dso->long_name); | 
					
						
							|  |  |  | 	ret += fprintf(fp, "%s, %sloaded, ", map_type__name[type], | 
					
						
							| 
									
										
										
										
											2012-11-20 10:51:02 +01:00
										 |  |  | 		       dso__loaded(dso, type) ? "" : "NOT "); | 
					
						
							| 
									
										
										
										
											2012-10-27 23:18:32 +02:00
										 |  |  | 	ret += dso__fprintf_buildid(dso, fp); | 
					
						
							|  |  |  | 	ret += fprintf(fp, ")\n"); | 
					
						
							|  |  |  | 	for (nd = rb_first(&dso->symbols[type]); nd; nd = rb_next(nd)) { | 
					
						
							|  |  |  | 		struct symbol *pos = rb_entry(nd, struct symbol, rb_node); | 
					
						
							|  |  |  | 		ret += symbol__fprintf(pos, fp); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } |