blob: ae0d4992dcf3297bbd6decf34b0f506c88b08d0b [file] [log] [blame]
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +00001/*
Richard Hendersonc49d1c32024-04-12 00:33:23 -07002* Helper to hexdump a buffer
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +00003 *
4 * Copyright (c) 2013 Red Hat, Inc.
5 * Copyright (c) 2013 Gerd Hoffmann <kraxel@redhat.com>
6 * Copyright (c) 2013 Peter Crosthwaite <peter.crosthwaite@xilinx.com>
7 * Copyright (c) 2013 Xilinx, Inc
8 *
9 * This work is licensed under the terms of the GNU GPL, version 2. See
10 * the COPYING file in the top-level directory.
11 *
12 * Contributions after 2012-01-13 are licensed under the terms of the
13 * GNU GPL, version 2 or (at your option) any later version.
14 */
15
Peter Maydellaafd7582016-01-29 17:49:55 +000016#include "qemu/osdep.h"
Marc-André Lureau415b7322022-03-23 19:57:32 +040017#include "qemu/cutils.h"
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +000018
Richard Henderson10e49272024-04-12 00:33:24 -070019static inline char hexdump_nibble(unsigned x)
20{
21 return (x < 10 ? '0' : 'a' - 10) + x;
22}
23
Richard Hendersonc49d1c32024-04-12 00:33:23 -070024GString *qemu_hexdump_line(GString *str, const void *vbuf, size_t len,
25 size_t unit_len, size_t block_len)
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +000026{
Richard Henderson53ee5f52024-04-12 00:33:22 -070027 const uint8_t *buf = vbuf;
Richard Hendersonc49d1c32024-04-12 00:33:23 -070028 size_t u, b;
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +000029
Richard Henderson53ee5f52024-04-12 00:33:22 -070030 if (str == NULL) {
31 /* Estimate the length of the output to avoid reallocs. */
Richard Hendersonc49d1c32024-04-12 00:33:23 -070032 size_t est = len * 2;
33 if (unit_len) {
34 est += len / unit_len;
35 }
36 if (block_len) {
37 est += len / block_len;
38 }
39 str = g_string_sized_new(est + 1);
Laurent Vivierbbb16902020-09-25 11:10:54 +020040 }
41
Richard Hendersonc49d1c32024-04-12 00:33:23 -070042 for (u = 0, b = 0; len; u++, b++, len--, buf++) {
Richard Henderson10e49272024-04-12 00:33:24 -070043 uint8_t c;
44
Richard Hendersonc49d1c32024-04-12 00:33:23 -070045 if (unit_len && u == unit_len) {
Richard Henderson53ee5f52024-04-12 00:33:22 -070046 g_string_append_c(str, ' ');
Richard Hendersonc49d1c32024-04-12 00:33:23 -070047 u = 0;
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +000048 }
Richard Hendersonc49d1c32024-04-12 00:33:23 -070049 if (block_len && b == block_len) {
50 g_string_append_c(str, ' ');
51 b = 0;
52 }
Richard Henderson10e49272024-04-12 00:33:24 -070053
54 c = *buf;
55 g_string_append_c(str, hexdump_nibble(c / 16));
56 g_string_append_c(str, hexdump_nibble(c % 16));
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +000057 }
Richard Henderson53ee5f52024-04-12 00:33:22 -070058
59 return str;
Laurent Vivierbbb16902020-09-25 11:10:54 +020060}
61
Richard Henderson13dfa932024-04-12 00:33:21 -070062static void asciidump_line(char *line, const void *bufptr, size_t len)
63{
64 const char *buf = bufptr;
65
66 for (size_t i = 0; i < len; i++) {
67 char c = buf[i];
68
69 if (c < ' ' || c > '~') {
70 c = '.';
71 }
72 *line++ = c;
73 }
74 *line = '\0';
75}
76
Richard Henderson53ee5f52024-04-12 00:33:22 -070077#define QEMU_HEXDUMP_LINE_BYTES 16
Richard Henderson13dfa932024-04-12 00:33:21 -070078#define QEMU_HEXDUMP_LINE_WIDTH \
79 (QEMU_HEXDUMP_LINE_BYTES * 2 + QEMU_HEXDUMP_LINE_BYTES / 4)
80
Laurent Vivierbbb16902020-09-25 11:10:54 +020081void qemu_hexdump(FILE *fp, const char *prefix,
82 const void *bufptr, size_t size)
83{
Richard Henderson53ee5f52024-04-12 00:33:22 -070084 g_autoptr(GString) str = g_string_sized_new(QEMU_HEXDUMP_LINE_WIDTH + 1);
Richard Henderson13dfa932024-04-12 00:33:21 -070085 char ascii[QEMU_HEXDUMP_LINE_BYTES + 1];
86 size_t b, len;
Laurent Vivierbbb16902020-09-25 11:10:54 +020087
Richard Henderson13dfa932024-04-12 00:33:21 -070088 for (b = 0; b < size; b += len) {
89 len = MIN(size - b, QEMU_HEXDUMP_LINE_BYTES);
90
Richard Henderson53ee5f52024-04-12 00:33:22 -070091 g_string_truncate(str, 0);
Richard Hendersonc49d1c32024-04-12 00:33:23 -070092 qemu_hexdump_line(str, bufptr + b, len, 1, 4);
Richard Henderson13dfa932024-04-12 00:33:21 -070093 asciidump_line(ascii, bufptr + b, len);
94
95 fprintf(fp, "%s: %04zx: %-*s %s\n",
Richard Henderson53ee5f52024-04-12 00:33:22 -070096 prefix, b, QEMU_HEXDUMP_LINE_WIDTH, str->str, ascii);
Laurent Vivierbbb16902020-09-25 11:10:54 +020097 }
98
Peter Crosthwaite6ff66f52013-03-15 16:41:58 +000099}