| /* Copyright (c) 2017 Google Inc. | 
 |  * See LICENSE for details. | 
 |  * | 
 |  * Utility functions. */ | 
 |  | 
 | #include <parlib/stdio.h> | 
 | #include <string.h> | 
 | #include <stdlib.h> | 
 | #include <unistd.h> | 
 | #include <errno.h> | 
 | #include <sys/fcntl.h> | 
 | #include <sys/types.h> | 
 | #include <sys/stat.h> | 
 | #include <vmm/util.h> | 
 | #include <vmm/vmm.h> | 
 |  | 
 | ssize_t cat(char *filename, void *where, size_t memsize) | 
 | { | 
 | 	int fd; | 
 | 	ssize_t amt, tot = 0; | 
 | 	struct stat buf; | 
 |  | 
 | 	fd = open(filename, O_RDONLY); | 
 | 	if (fd < 0) { | 
 | 		fprintf(stderr, "Can't open %s: %r\n", filename); | 
 | 		return -1; | 
 | 	} | 
 |  | 
 | 	if (fstat(fd, &buf) < 0) { | 
 | 		fprintf(stderr, "Can't stat %s: %r\n", filename); | 
 | 		close(fd); | 
 | 		return -1; | 
 | 	} | 
 |  | 
 | 	if (buf.st_size > memsize) { | 
 | 		fprintf(stderr, | 
 | 		        "file is %d bytes, but we only have %d bytes to place it\n", | 
 | 		        buf.st_size, memsize); | 
 | 		errno = ENOMEM; | 
 | 		close(fd); | 
 | 		return -1; | 
 | 	} | 
 |  | 
 | 	while (tot < buf.st_size) { | 
 | 		amt = read(fd, where, buf.st_size - tot); | 
 | 		if (amt < 0) { | 
 | 			tot = -1; | 
 | 			break; | 
 | 		} | 
 | 		if (amt == 0) | 
 | 			break; | 
 | 		where += amt; | 
 | 		tot += amt; | 
 | 	} | 
 |  | 
 | 	close(fd); | 
 | 	return tot; | 
 | } | 
 |  | 
 | void backtrace_guest_thread(FILE *f, struct guest_thread *vm_thread) | 
 | { | 
 | 	struct vm_trapframe *vm_tf = gth_to_vmtf(vm_thread); | 
 | 	uintptr_t guest_pc, guest_fp, host_fp; | 
 | 	uintptr_t frame[2]; | 
 |  | 
 | 	guest_pc = vm_tf->tf_rip; | 
 | 	guest_fp = vm_tf->tf_rbp; | 
 |  | 
 | 	/* The BT should work for any code using frame pointers.  Most of the | 
 | 	 * time, this will be vmlinux, and calling it that helps our backtrace. | 
 | 	 * This spits out the format that is parsed by bt-akaros.sh. */ | 
 | 	fprintf(f, "Backtracing guest, vmlinux is assumed, but check addrs\n"); | 
 | 	for (int i = 1; i < 30; i++) { | 
 | 		fprintf(f, "#%02d Addr %p is in vmlinux at offset %p\n", i, | 
 | 			guest_pc, guest_pc); | 
 | 		if (!guest_fp) | 
 | 			break; | 
 | 		if (gva2gpa(vm_thread, guest_fp, &host_fp)) | 
 | 			break; | 
 | 		memcpy(frame, (void*)host_fp, 2 * sizeof(uintptr_t)); | 
 | 		guest_pc = frame[1]; | 
 | 		guest_fp = frame[0]; | 
 | 	} | 
 | } |