|  | // SPDX-License-Identifier: GPL-2.0-only | 
|  | /* | 
|  | * intel_pt_log.c: Intel Processor Trace support | 
|  | * Copyright (c) 2013-2014, Intel Corporation. | 
|  | */ | 
|  |  | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <stdint.h> | 
|  | #include <inttypes.h> | 
|  | #include <stdarg.h> | 
|  | #include <stdbool.h> | 
|  | #include <string.h> | 
|  |  | 
|  | #include <linux/zalloc.h> | 
|  | #include <linux/kernel.h> | 
|  |  | 
|  | #include "intel-pt-log.h" | 
|  | #include "intel-pt-insn-decoder.h" | 
|  |  | 
|  | #include "intel-pt-pkt-decoder.h" | 
|  |  | 
|  | #define MAX_LOG_NAME 256 | 
|  |  | 
|  | #define DFLT_BUF_SZ	(16 * 1024) | 
|  |  | 
|  | struct log_buf { | 
|  | char			*buf; | 
|  | size_t			buf_sz; | 
|  | size_t			head; | 
|  | bool			wrapped; | 
|  | FILE			*backend; | 
|  | }; | 
|  |  | 
|  | static FILE *f; | 
|  | static char log_name[MAX_LOG_NAME]; | 
|  | bool intel_pt_enable_logging; | 
|  | static bool intel_pt_dump_log_on_error; | 
|  | static unsigned int intel_pt_log_on_error_size; | 
|  | static struct log_buf log_buf; | 
|  |  | 
|  | void *intel_pt_log_fp(void) | 
|  | { | 
|  | return f; | 
|  | } | 
|  |  | 
|  | void intel_pt_log_enable(bool dump_log_on_error, unsigned int log_on_error_size) | 
|  | { | 
|  | intel_pt_enable_logging = true; | 
|  | intel_pt_dump_log_on_error = dump_log_on_error; | 
|  | intel_pt_log_on_error_size = log_on_error_size; | 
|  | } | 
|  |  | 
|  | void intel_pt_log_disable(void) | 
|  | { | 
|  | if (f) | 
|  | fflush(f); | 
|  | intel_pt_enable_logging = false; | 
|  | } | 
|  |  | 
|  | void intel_pt_log_set_name(const char *name) | 
|  | { | 
|  | strncpy(log_name, name, MAX_LOG_NAME - 5); | 
|  | strcat(log_name, ".log"); | 
|  | } | 
|  |  | 
|  | static void intel_pt_print_data(const unsigned char *buf, int len, uint64_t pos, | 
|  | int indent) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < indent; i++) | 
|  | fprintf(f, " "); | 
|  |  | 
|  | fprintf(f, "  %08" PRIx64 ": ", pos); | 
|  | for (i = 0; i < len; i++) | 
|  | fprintf(f, " %02x", buf[i]); | 
|  | for (; i < 16; i++) | 
|  | fprintf(f, "   "); | 
|  | fprintf(f, " "); | 
|  | } | 
|  |  | 
|  | static void intel_pt_print_no_data(uint64_t pos, int indent) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < indent; i++) | 
|  | fprintf(f, " "); | 
|  |  | 
|  | fprintf(f, "  %08" PRIx64 ": ", pos); | 
|  | for (i = 0; i < 16; i++) | 
|  | fprintf(f, "   "); | 
|  | fprintf(f, " "); | 
|  | } | 
|  |  | 
|  | static ssize_t log_buf__write(void *cookie, const char *buf, size_t size) | 
|  | { | 
|  | struct log_buf *b = cookie; | 
|  | size_t sz = size; | 
|  |  | 
|  | if (!b->buf) | 
|  | return size; | 
|  |  | 
|  | while (sz) { | 
|  | size_t space = b->buf_sz - b->head; | 
|  | size_t n = min(space, sz); | 
|  |  | 
|  | memcpy(b->buf + b->head, buf, n); | 
|  | sz -= n; | 
|  | buf += n; | 
|  | b->head += n; | 
|  | if (sz && b->head >= b->buf_sz) { | 
|  | b->head = 0; | 
|  | b->wrapped = true; | 
|  | } | 
|  | } | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static int log_buf__close(void *cookie) | 
|  | { | 
|  | struct log_buf *b = cookie; | 
|  |  | 
|  | zfree(&b->buf); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static FILE *log_buf__open(struct log_buf *b, FILE *backend, unsigned int sz) | 
|  | { | 
|  | cookie_io_functions_t fns = { | 
|  | .write = log_buf__write, | 
|  | .close = log_buf__close, | 
|  | }; | 
|  | FILE *file; | 
|  |  | 
|  | memset(b, 0, sizeof(*b)); | 
|  | b->buf_sz = sz; | 
|  | b->buf = malloc(b->buf_sz); | 
|  | b->backend = backend; | 
|  | file = fopencookie(b, "a", fns); | 
|  | if (!file) | 
|  | zfree(&b->buf); | 
|  | return file; | 
|  | } | 
|  |  | 
|  | static bool remove_first_line(const char **p, size_t *n) | 
|  | { | 
|  | for (; *n && **p != '\n'; ++*p, --*n) | 
|  | ; | 
|  | if (*n) { | 
|  | *p += 1; | 
|  | *n -= 1; | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | static void write_lines(const char *p, size_t n, FILE *fp, bool *remove_first) | 
|  | { | 
|  | if (*remove_first) | 
|  | *remove_first = !remove_first_line(&p, &n); | 
|  | fwrite(p, n, 1, fp); | 
|  | } | 
|  |  | 
|  | static void log_buf__dump(struct log_buf *b) | 
|  | { | 
|  | bool remove_first = false; | 
|  |  | 
|  | if (!b->buf) | 
|  | return; | 
|  |  | 
|  | fflush(f); /* Could update b->head and b->wrapped */ | 
|  | fprintf(b->backend, "Dumping debug log buffer\n"); | 
|  | if (b->wrapped) { | 
|  | remove_first = true; | 
|  | write_lines(b->buf + b->head, b->buf_sz - b->head, b->backend, &remove_first); | 
|  | } | 
|  | write_lines(b->buf, b->head, b->backend, &remove_first); | 
|  | fprintf(b->backend, "End of debug log buffer dump\n"); | 
|  |  | 
|  | b->head = 0; | 
|  | b->wrapped = false; | 
|  | } | 
|  |  | 
|  | void intel_pt_log_dump_buf(void) | 
|  | { | 
|  | log_buf__dump(&log_buf); | 
|  | } | 
|  |  | 
|  | static int intel_pt_log_open(void) | 
|  | { | 
|  | if (!intel_pt_enable_logging) | 
|  | return -1; | 
|  |  | 
|  | if (f) | 
|  | return 0; | 
|  |  | 
|  | if (log_name[0]) | 
|  | f = fopen(log_name, "w+"); | 
|  | else | 
|  | f = stdout; | 
|  | if (f && intel_pt_dump_log_on_error) | 
|  | f = log_buf__open(&log_buf, f, intel_pt_log_on_error_size); | 
|  | if (!f) { | 
|  | intel_pt_enable_logging = false; | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | void __intel_pt_log_packet(const struct intel_pt_pkt *packet, int pkt_len, | 
|  | uint64_t pos, const unsigned char *buf) | 
|  | { | 
|  | char desc[INTEL_PT_PKT_DESC_MAX]; | 
|  |  | 
|  | if (intel_pt_log_open()) | 
|  | return; | 
|  |  | 
|  | intel_pt_print_data(buf, pkt_len, pos, 0); | 
|  | intel_pt_pkt_desc(packet, desc, INTEL_PT_PKT_DESC_MAX); | 
|  | fprintf(f, "%s\n", desc); | 
|  | } | 
|  |  | 
|  | void __intel_pt_log_insn(struct intel_pt_insn *intel_pt_insn, uint64_t ip) | 
|  | { | 
|  | char desc[INTEL_PT_INSN_DESC_MAX]; | 
|  | size_t len = intel_pt_insn->length; | 
|  |  | 
|  | if (intel_pt_log_open()) | 
|  | return; | 
|  |  | 
|  | if (len > INTEL_PT_INSN_BUF_SZ) | 
|  | len = INTEL_PT_INSN_BUF_SZ; | 
|  | intel_pt_print_data(intel_pt_insn->buf, len, ip, 8); | 
|  | if (intel_pt_insn_desc(intel_pt_insn, desc, INTEL_PT_INSN_DESC_MAX) > 0) | 
|  | fprintf(f, "%s\n", desc); | 
|  | else | 
|  | fprintf(f, "Bad instruction!\n"); | 
|  | } | 
|  |  | 
|  | void __intel_pt_log_insn_no_data(struct intel_pt_insn *intel_pt_insn, | 
|  | uint64_t ip) | 
|  | { | 
|  | char desc[INTEL_PT_INSN_DESC_MAX]; | 
|  |  | 
|  | if (intel_pt_log_open()) | 
|  | return; | 
|  |  | 
|  | intel_pt_print_no_data(ip, 8); | 
|  | if (intel_pt_insn_desc(intel_pt_insn, desc, INTEL_PT_INSN_DESC_MAX) > 0) | 
|  | fprintf(f, "%s\n", desc); | 
|  | else | 
|  | fprintf(f, "Bad instruction!\n"); | 
|  | } | 
|  |  | 
|  | void __intel_pt_log(const char *fmt, ...) | 
|  | { | 
|  | va_list args; | 
|  |  | 
|  | if (intel_pt_log_open()) | 
|  | return; | 
|  |  | 
|  | va_start(args, fmt); | 
|  | vfprintf(f, fmt, args); | 
|  | va_end(args); | 
|  | } |