|  | /* 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]; | 
|  | } | 
|  | } |