db_disasm.c revision 1.9 1 /* $NetBSD: db_disasm.c,v 1.9 2020/07/08 03:45:13 ryo Exp $ */
2
3 /*
4 * Copyright (c) 2017 Ryo Shimizu <ryo (at) nerv.org>
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
20 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
24 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
25 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26 * POSSIBILITY OF SUCH DAMAGE.
27 */
28
29 #include <sys/cdefs.h>
30 __KERNEL_RCSID(0, "$NetBSD: db_disasm.c,v 1.9 2020/07/08 03:45:13 ryo Exp $");
31
32 #ifdef _KERNEL_OPT
33 #include "opt_compat_netbsd32.h"
34 #endif
35
36 #include <sys/param.h>
37 #include <machine/db_machdep.h>
38 #include <ddb/db_interface.h>
39 #include <ddb/db_sym.h>
40 #include <ddb/db_output.h>
41 #include <ddb/db_access.h>
42 #include <ddb/db_user.h>
43
44 #include <aarch64/cpufunc.h>
45 #include <aarch64/machdep.h>
46 #include <arch/aarch64/aarch64/disasm.h>
47
48 static uint32_t
49 db_disasm_readword(uintptr_t address)
50 {
51 return db_get_value(address, sizeof(uint32_t), false);
52 }
53
54 static void
55 db_disasm_printaddr(uintptr_t address)
56 {
57 db_printf("%lx <", address);
58 db_printsym((db_addr_t)address, DB_STGY_ANY, db_printf);
59 db_printf(">");
60 }
61
62 static const disasm_interface_t db_disasm_interface = {
63 .di_readword = db_disasm_readword,
64 .di_printaddr = db_disasm_printaddr,
65 .di_printf = db_printf
66 };
67
68 db_addr_t
69 db_disasm(db_addr_t loc, bool altfmt)
70 {
71 return disasm(&db_disasm_interface, loc);
72 }
73
74
75 static char *strdisasm_ptr;
76 static char strdisasm_buf[256];
77
78 static uint32_t
79 strdisasm_readword(uintptr_t address)
80 {
81 /*
82 * if it cannot be read due to a EFAULT etc.,
83 * ignores the error and returns 0
84 */
85 uint32_t word = 0;
86
87 switch (aarch64_addressspace((vaddr_t)address)) {
88 case AARCH64_ADDRSPACE_UPPER:
89 kcopy((void*)address, &word, sizeof(word));
90 break;
91 case AARCH64_ADDRSPACE_LOWER:
92 ufetch_32((uint32_t *)address, &word);
93 break;
94 default:
95 break;
96 }
97
98 return word;
99 }
100
101 static void __printflike(1, 2)
102 strdisasm_printf(const char *fmt, ...)
103 {
104 va_list ap;
105 int len;
106
107 /* calculation spaces to append a string */
108 len = strdisasm_buf + sizeof(strdisasm_buf) - strdisasm_ptr;
109
110 va_start(ap, fmt);
111 len = vsnprintf(strdisasm_ptr, len, fmt, ap);
112 va_end(ap);
113
114 strdisasm_ptr += len;
115 }
116
117 static void
118 strdisasm_printaddr(uintptr_t address)
119 {
120 strdisasm_printf("0x%lx", address);
121 }
122
123 static const disasm_interface_t strdisasm_interface = {
124 .di_readword = strdisasm_readword,
125 .di_printaddr = strdisasm_printaddr,
126 .di_printf = strdisasm_printf
127 };
128
129 const char *
130 strdisasm(vaddr_t pc, uint64_t spsr)
131 {
132 #ifdef COMPAT_NETBSD32
133 if (spsr & SPSR_A32) {
134 uint32_t insn = 0;
135 int size;
136 const char *arch = (spsr & SPSR_A32_T) ? "T32" : "A32";
137
138 size = fetch_arm_insn(pc, spsr, &insn);
139 if (size != 2)
140 size = 4;
141 snprintf(strdisasm_buf, sizeof(strdisasm_buf),
142 ".insn 0x%0*x (%s)", size * 2, insn, arch);
143 } else
144 #endif /* COMPAT_NETBSD32 */
145 {
146 char *p;
147
148 /* disasm an aarch64 instruction */
149 strdisasm_ptr = strdisasm_buf;
150 disasm(&strdisasm_interface, (db_addr_t)pc);
151
152 /* replace tab to space, and chomp '\n' */
153 for (p = strdisasm_buf; *p != '\0'; p++) {
154 if (*p == '\t')
155 *p = ' ';
156 }
157 if ((p > strdisasm_buf) && (p[-1] == '\n'))
158 p[-1] = '\0';
159 }
160
161 return strdisasm_buf;
162 }
163