Linux kernel mirror (for testing)
git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel
os
linux
1// SPDX-License-Identifier: GPL-2.0-or-later
2/*
3 * Copyright (C) 2015-2017 Josh Poimboeuf <jpoimboe@redhat.com>
4 */
5
6#define _GNU_SOURCE
7#include <fnmatch.h>
8
9#include <objtool/arch.h>
10#include <objtool/check.h>
11#include <objtool/disas.h>
12#include <objtool/special.h>
13#include <objtool/warn.h>
14
15#include <bfd.h>
16#include <linux/string.h>
17#include <tools/dis-asm-compat.h>
18
19/*
20 * Size of the buffer for storing the result of disassembling
21 * a single instruction.
22 */
23#define DISAS_RESULT_SIZE 1024
24
25struct disas_context {
26 struct objtool_file *file;
27 struct instruction *insn;
28 bool alt_applied;
29 char result[DISAS_RESULT_SIZE];
30 disassembler_ftype disassembler;
31 struct disassemble_info info;
32};
33
34/*
35 * Maximum number of alternatives
36 */
37#define DISAS_ALT_MAX 5
38
39/*
40 * Maximum number of instructions per alternative
41 */
42#define DISAS_ALT_INSN_MAX 50
43
44/*
45 * Information to disassemble an alternative
46 */
47struct disas_alt {
48 struct instruction *orig_insn; /* original instruction */
49 struct alternative *alt; /* alternative or NULL if default code */
50 char *name; /* name for this alternative */
51 int width; /* formatting width */
52 struct {
53 char *str; /* instruction string */
54 int offset; /* instruction offset */
55 int nops; /* number of nops */
56 } insn[DISAS_ALT_INSN_MAX]; /* alternative instructions */
57 int insn_idx; /* index of the next instruction to print */
58};
59
60#define DALT_DEFAULT(dalt) (!(dalt)->alt)
61#define DALT_INSN(dalt) (DALT_DEFAULT(dalt) ? (dalt)->orig_insn : (dalt)->alt->insn)
62#define DALT_GROUP(dalt) (DALT_INSN(dalt)->alt_group)
63#define DALT_ALTID(dalt) ((dalt)->orig_insn->offset)
64
65#define ALT_FLAGS_SHIFT 16
66#define ALT_FLAG_NOT (1 << 0)
67#define ALT_FLAG_DIRECT_CALL (1 << 1)
68#define ALT_FEATURE_MASK ((1 << ALT_FLAGS_SHIFT) - 1)
69
70static int alt_feature(unsigned int ft_flags)
71{
72 return (ft_flags & ALT_FEATURE_MASK);
73}
74
75static int alt_flags(unsigned int ft_flags)
76{
77 return (ft_flags >> ALT_FLAGS_SHIFT);
78}
79
80/*
81 * Wrapper around asprintf() to allocate and format a string.
82 * Return the allocated string or NULL on error.
83 */
84static char *strfmt(const char *fmt, ...)
85{
86 va_list ap;
87 char *str;
88 int rv;
89
90 va_start(ap, fmt);
91 rv = vasprintf(&str, fmt, ap);
92 va_end(ap);
93
94 return rv == -1 ? NULL : str;
95}
96
97static int sprint_name(char *str, const char *name, unsigned long offset)
98{
99 int len;
100
101 if (offset)
102 len = sprintf(str, "%s+0x%lx", name, offset);
103 else
104 len = sprintf(str, "%s", name);
105
106 return len;
107}
108
109#define DINFO_FPRINTF(dinfo, ...) \
110 ((*(dinfo)->fprintf_func)((dinfo)->stream, __VA_ARGS__))
111
112static int disas_result_fprintf(struct disas_context *dctx,
113 const char *fmt, va_list ap)
114{
115 char *buf = dctx->result;
116 int avail, len;
117
118 len = strlen(buf);
119 if (len >= DISAS_RESULT_SIZE - 1) {
120 WARN_FUNC(dctx->insn->sec, dctx->insn->offset,
121 "disassembly buffer is full");
122 return -1;
123 }
124 avail = DISAS_RESULT_SIZE - len;
125
126 len = vsnprintf(buf + len, avail, fmt, ap);
127 if (len < 0 || len >= avail) {
128 WARN_FUNC(dctx->insn->sec, dctx->insn->offset,
129 "disassembly buffer is truncated");
130 return -1;
131 }
132
133 return 0;
134}
135
136static int disas_fprintf(void *stream, const char *fmt, ...)
137{
138 va_list arg;
139 int rv;
140
141 va_start(arg, fmt);
142 rv = disas_result_fprintf(stream, fmt, arg);
143 va_end(arg);
144
145 return rv;
146}
147
148/*
149 * For init_disassemble_info_compat().
150 */
151static int disas_fprintf_styled(void *stream,
152 enum disassembler_style style,
153 const char *fmt, ...)
154{
155 va_list arg;
156 int rv;
157
158 va_start(arg, fmt);
159 rv = disas_result_fprintf(stream, fmt, arg);
160 va_end(arg);
161
162 return rv;
163}
164
165static void disas_print_addr_sym(struct section *sec, struct symbol *sym,
166 bfd_vma addr, struct disassemble_info *dinfo)
167{
168 char symstr[1024];
169 char *str;
170
171 if (sym) {
172 sprint_name(symstr, sym->name, addr - sym->offset);
173 DINFO_FPRINTF(dinfo, "0x%lx <%s>", addr, symstr);
174 } else {
175 str = offstr(sec, addr);
176 DINFO_FPRINTF(dinfo, "0x%lx <%s>", addr, str);
177 free(str);
178 }
179}
180
181static bool disas_print_addr_alt(bfd_vma addr, struct disassemble_info *dinfo)
182{
183 struct disas_context *dctx = dinfo->application_data;
184 struct instruction *orig_first_insn;
185 struct alt_group *alt_group;
186 unsigned long offset;
187 struct symbol *sym;
188
189 /*
190 * Check if we are processing an alternative at the original
191 * instruction address (i.e. if alt_applied is true) and if
192 * we are referencing an address inside the alternative.
193 *
194 * For example, this happens if there is a branch inside an
195 * alternative. In that case, the address should be updated
196 * to a reference inside the original instruction flow.
197 */
198 if (!dctx->alt_applied)
199 return false;
200
201 alt_group = dctx->insn->alt_group;
202 if (!alt_group || !alt_group->orig_group ||
203 addr < alt_group->first_insn->offset ||
204 addr > alt_group->last_insn->offset)
205 return false;
206
207 orig_first_insn = alt_group->orig_group->first_insn;
208 offset = addr - alt_group->first_insn->offset;
209
210 addr = orig_first_insn->offset + offset;
211 sym = orig_first_insn->sym;
212
213 disas_print_addr_sym(orig_first_insn->sec, sym, addr, dinfo);
214
215 return true;
216}
217
218static void disas_print_addr_noreloc(bfd_vma addr,
219 struct disassemble_info *dinfo)
220{
221 struct disas_context *dctx = dinfo->application_data;
222 struct instruction *insn = dctx->insn;
223 struct symbol *sym = NULL;
224
225 if (disas_print_addr_alt(addr, dinfo))
226 return;
227
228 if (insn->sym && addr >= insn->sym->offset &&
229 addr < insn->sym->offset + insn->sym->len) {
230 sym = insn->sym;
231 }
232
233 disas_print_addr_sym(insn->sec, sym, addr, dinfo);
234}
235
236static void disas_print_addr_reloc(bfd_vma addr, struct disassemble_info *dinfo)
237{
238 struct disas_context *dctx = dinfo->application_data;
239 struct instruction *insn = dctx->insn;
240 unsigned long offset;
241 struct reloc *reloc;
242 char symstr[1024];
243 char *str;
244
245 reloc = find_reloc_by_dest_range(dctx->file->elf, insn->sec,
246 insn->offset, insn->len);
247 if (!reloc) {
248 /*
249 * There is no relocation for this instruction although
250 * the address to resolve points to the next instruction.
251 * So this is an effective reference to the next IP, for
252 * example: "lea 0x0(%rip),%rdi". The kernel can reference
253 * the next IP with _THIS_IP_ macro.
254 */
255 DINFO_FPRINTF(dinfo, "0x%lx <_THIS_IP_>", addr);
256 return;
257 }
258
259 offset = arch_insn_adjusted_addend(insn, reloc);
260
261 /*
262 * If the relocation symbol is a section name (for example ".bss")
263 * then we try to further resolve the name.
264 */
265 if (reloc->sym->type == STT_SECTION) {
266 str = offstr(reloc->sym->sec, reloc->sym->offset + offset);
267 DINFO_FPRINTF(dinfo, "0x%lx <%s>", addr, str);
268 free(str);
269 } else {
270 sprint_name(symstr, reloc->sym->name, offset);
271 DINFO_FPRINTF(dinfo, "0x%lx <%s>", addr, symstr);
272 }
273}
274
275/*
276 * Resolve an address into a "<symbol>+<offset>" string.
277 */
278static void disas_print_address(bfd_vma addr, struct disassemble_info *dinfo)
279{
280 struct disas_context *dctx = dinfo->application_data;
281 struct instruction *insn = dctx->insn;
282 struct instruction *jump_dest;
283 struct symbol *sym;
284 bool is_reloc;
285
286 /*
287 * If the instruction is a call/jump and it references a
288 * destination then this is likely the address we are looking
289 * up. So check it first.
290 */
291 jump_dest = insn->jump_dest;
292 if (jump_dest && jump_dest->sym && jump_dest->offset == addr) {
293 if (!disas_print_addr_alt(addr, dinfo))
294 disas_print_addr_sym(jump_dest->sec, jump_dest->sym,
295 addr, dinfo);
296 return;
297 }
298
299 /*
300 * If the address points to the next instruction then there is
301 * probably a relocation. It can be a false positive when the
302 * current instruction is referencing the address of the next
303 * instruction. This particular case will be handled in
304 * disas_print_addr_reloc().
305 */
306 is_reloc = (addr == insn->offset + insn->len);
307
308 /*
309 * The call destination offset can be the address we are looking
310 * up, or 0 if there is a relocation.
311 */
312 sym = insn_call_dest(insn);
313 if (sym && (sym->offset == addr || (sym->offset == 0 && is_reloc))) {
314 DINFO_FPRINTF(dinfo, "0x%lx <%s>", addr, sym->name);
315 return;
316 }
317
318 if (!is_reloc)
319 disas_print_addr_noreloc(addr, dinfo);
320 else
321 disas_print_addr_reloc(addr, dinfo);
322}
323
324/*
325 * Initialize disassemble info arch, mach (32 or 64-bit) and options.
326 */
327int disas_info_init(struct disassemble_info *dinfo,
328 int arch, int mach32, int mach64,
329 const char *options)
330{
331 struct disas_context *dctx = dinfo->application_data;
332 struct objtool_file *file = dctx->file;
333
334 dinfo->arch = arch;
335
336 switch (file->elf->ehdr.e_ident[EI_CLASS]) {
337 case ELFCLASS32:
338 dinfo->mach = mach32;
339 break;
340 case ELFCLASS64:
341 dinfo->mach = mach64;
342 break;
343 default:
344 return -1;
345 }
346
347 dinfo->disassembler_options = options;
348
349 return 0;
350}
351
352struct disas_context *disas_context_create(struct objtool_file *file)
353{
354 struct disas_context *dctx;
355 struct disassemble_info *dinfo;
356 int err;
357
358 dctx = malloc(sizeof(*dctx));
359 if (!dctx) {
360 WARN("failed to allocate disassembly context");
361 return NULL;
362 }
363
364 dctx->file = file;
365 dinfo = &dctx->info;
366
367 init_disassemble_info_compat(dinfo, dctx,
368 disas_fprintf, disas_fprintf_styled);
369
370 dinfo->read_memory_func = buffer_read_memory;
371 dinfo->print_address_func = disas_print_address;
372 dinfo->application_data = dctx;
373
374 /*
375 * bfd_openr() is not used to avoid doing ELF data processing
376 * and caching that has already being done. Here, we just need
377 * to identify the target file so we call an arch specific
378 * function to fill some disassemble info (arch, mach).
379 */
380
381 dinfo->arch = bfd_arch_unknown;
382 dinfo->mach = 0;
383
384 err = arch_disas_info_init(dinfo);
385 if (err || dinfo->arch == bfd_arch_unknown || dinfo->mach == 0) {
386 WARN("failed to init disassembly arch");
387 goto error;
388 }
389
390 dinfo->endian = (file->elf->ehdr.e_ident[EI_DATA] == ELFDATA2MSB) ?
391 BFD_ENDIAN_BIG : BFD_ENDIAN_LITTLE;
392
393 disassemble_init_for_target(dinfo);
394
395 dctx->disassembler = disassembler(dinfo->arch,
396 dinfo->endian == BFD_ENDIAN_BIG,
397 dinfo->mach, NULL);
398 if (!dctx->disassembler) {
399 WARN("failed to create disassembler function");
400 goto error;
401 }
402
403 return dctx;
404
405error:
406 free(dctx);
407 return NULL;
408}
409
410void disas_context_destroy(struct disas_context *dctx)
411{
412 free(dctx);
413}
414
415char *disas_result(struct disas_context *dctx)
416{
417 return dctx->result;
418}
419
420#define DISAS_INSN_OFFSET_SPACE 10
421#define DISAS_INSN_SPACE 60
422
423#define DISAS_PRINSN(dctx, insn, depth) \
424 disas_print_insn(stdout, dctx, insn, depth, "\n")
425
426/*
427 * Print a message in the instruction flow. If sec is not NULL then the
428 * address at the section offset is printed in addition of the message,
429 * otherwise only the message is printed.
430 */
431static int disas_vprint(FILE *stream, struct section *sec, unsigned long offset,
432 int depth, const char *format, va_list ap)
433{
434 const char *addr_str;
435 int i, n;
436 int len;
437
438 len = sym_name_max_len + DISAS_INSN_OFFSET_SPACE;
439 if (depth < 0) {
440 len += depth;
441 depth = 0;
442 }
443
444 n = 0;
445
446 if (sec) {
447 addr_str = offstr(sec, offset);
448 n += fprintf(stream, "%6lx: %-*s ", offset, len, addr_str);
449 free((char *)addr_str);
450 } else {
451 len += DISAS_INSN_OFFSET_SPACE + 1;
452 n += fprintf(stream, "%-*s", len, "");
453 }
454
455 /* print vertical bars to show the code flow */
456 for (i = 0; i < depth; i++)
457 n += fprintf(stream, "| ");
458
459 if (format)
460 n += vfprintf(stream, format, ap);
461
462 return n;
463}
464
465static int disas_print(FILE *stream, struct section *sec, unsigned long offset,
466 int depth, const char *format, ...)
467{
468 va_list args;
469 int len;
470
471 va_start(args, format);
472 len = disas_vprint(stream, sec, offset, depth, format, args);
473 va_end(args);
474
475 return len;
476}
477
478/*
479 * Print a message in the instruction flow. If insn is not NULL then
480 * the instruction address is printed in addition of the message,
481 * otherwise only the message is printed. In all cases, the instruction
482 * itself is not printed.
483 */
484void disas_print_info(FILE *stream, struct instruction *insn, int depth,
485 const char *format, ...)
486{
487 struct section *sec;
488 unsigned long off;
489 va_list args;
490
491 if (insn) {
492 sec = insn->sec;
493 off = insn->offset;
494 } else {
495 sec = NULL;
496 off = 0;
497 }
498
499 va_start(args, format);
500 disas_vprint(stream, sec, off, depth, format, args);
501 va_end(args);
502}
503
504/*
505 * Print an instruction address (offset and function), the instruction itself
506 * and an optional message.
507 */
508void disas_print_insn(FILE *stream, struct disas_context *dctx,
509 struct instruction *insn, int depth,
510 const char *format, ...)
511{
512 char fake_nop_insn[32];
513 const char *insn_str;
514 bool fake_nop;
515 va_list args;
516 int len;
517
518 /*
519 * Alternative can insert a fake nop, sometimes with no
520 * associated section so nothing to disassemble.
521 */
522 fake_nop = (!insn->sec && insn->type == INSN_NOP);
523 if (fake_nop) {
524 snprintf(fake_nop_insn, 32, "<fake nop> (%d bytes)", insn->len);
525 insn_str = fake_nop_insn;
526 } else {
527 disas_insn(dctx, insn);
528 insn_str = disas_result(dctx);
529 }
530
531 /* print the instruction */
532 len = (depth + 1) * 2 < DISAS_INSN_SPACE ? DISAS_INSN_SPACE - (depth+1) * 2 : 1;
533 disas_print_info(stream, insn, depth, "%-*s", len, insn_str);
534
535 /* print message if any */
536 if (!format)
537 return;
538
539 if (strcmp(format, "\n") == 0) {
540 fprintf(stream, "\n");
541 return;
542 }
543
544 fprintf(stream, " - ");
545 va_start(args, format);
546 vfprintf(stream, format, args);
547 va_end(args);
548}
549
550/*
551 * Disassemble a single instruction. Return the size of the instruction.
552 *
553 * If alt_applied is true then insn should be an instruction from of an
554 * alternative (i.e. insn->alt_group != NULL), and it is disassembled
555 * at the location of the original code it is replacing. When the
556 * instruction references any address inside the alternative then
557 * these references will be re-adjusted to replace the original code.
558 */
559static size_t disas_insn_common(struct disas_context *dctx,
560 struct instruction *insn,
561 bool alt_applied)
562{
563 disassembler_ftype disasm = dctx->disassembler;
564 struct disassemble_info *dinfo = &dctx->info;
565
566 dctx->insn = insn;
567 dctx->alt_applied = alt_applied;
568 dctx->result[0] = '\0';
569
570 if (insn->type == INSN_NOP) {
571 DINFO_FPRINTF(dinfo, "nop%d", insn->len);
572 return insn->len;
573 }
574
575 /*
576 * Set the disassembler buffer to read data from the section
577 * containing the instruction to disassemble.
578 */
579 dinfo->buffer = insn->sec->data->d_buf;
580 dinfo->buffer_vma = 0;
581 dinfo->buffer_length = insn->sec->sh.sh_size;
582
583 return disasm(insn->offset, &dctx->info);
584}
585
586size_t disas_insn(struct disas_context *dctx, struct instruction *insn)
587{
588 return disas_insn_common(dctx, insn, false);
589}
590
591static size_t disas_insn_alt(struct disas_context *dctx,
592 struct instruction *insn)
593{
594 return disas_insn_common(dctx, insn, true);
595}
596
597static struct instruction *next_insn_same_alt(struct objtool_file *file,
598 struct alt_group *alt_grp,
599 struct instruction *insn)
600{
601 if (alt_grp->last_insn == insn || alt_grp->nop == insn)
602 return NULL;
603
604 return next_insn_same_sec(file, insn);
605}
606
607#define alt_for_each_insn(file, alt_grp, insn) \
608 for (insn = alt_grp->first_insn; \
609 insn; \
610 insn = next_insn_same_alt(file, alt_grp, insn))
611
612/*
613 * Provide a name for the type of alternatives present at the
614 * specified instruction.
615 *
616 * An instruction can have alternatives with different types, for
617 * example alternative instructions and an exception table. In that
618 * case the name for the alternative instructions type is used.
619 *
620 * Return NULL if the instruction as no alternative.
621 */
622const char *disas_alt_type_name(struct instruction *insn)
623{
624 struct alternative *alt;
625 const char *name;
626
627 name = NULL;
628 for (alt = insn->alts; alt; alt = alt->next) {
629 if (alt->type == ALT_TYPE_INSTRUCTIONS) {
630 name = "alternative";
631 break;
632 }
633
634 switch (alt->type) {
635 case ALT_TYPE_EX_TABLE:
636 name = "ex_table";
637 break;
638 case ALT_TYPE_JUMP_TABLE:
639 name = "jump_table";
640 break;
641 default:
642 name = "unknown";
643 break;
644 }
645 }
646
647 return name;
648}
649
650/*
651 * Provide a name for an alternative.
652 */
653char *disas_alt_name(struct alternative *alt)
654{
655 char pfx[4] = { 0 };
656 char *str = NULL;
657 const char *name;
658 int feature;
659 int flags;
660 int num;
661
662 switch (alt->type) {
663
664 case ALT_TYPE_EX_TABLE:
665 str = strdup("EXCEPTION");
666 break;
667
668 case ALT_TYPE_JUMP_TABLE:
669 str = strdup("JUMP");
670 break;
671
672 case ALT_TYPE_INSTRUCTIONS:
673 /*
674 * This is a non-default group alternative. Create a name
675 * based on the feature and flags associated with this
676 * alternative. Use either the feature name (it is available)
677 * or the feature number. And add a prefix to show the flags
678 * used.
679 *
680 * Prefix flags characters:
681 *
682 * '!' alternative used when feature not enabled
683 * '+' direct call alternative
684 * '?' unknown flag
685 */
686
687 if (!alt->insn->alt_group)
688 return NULL;
689
690 feature = alt->insn->alt_group->feature;
691 num = alt_feature(feature);
692 flags = alt_flags(feature);
693 str = pfx;
694
695 if (flags & ~(ALT_FLAG_NOT | ALT_FLAG_DIRECT_CALL))
696 *str++ = '?';
697 if (flags & ALT_FLAG_DIRECT_CALL)
698 *str++ = '+';
699 if (flags & ALT_FLAG_NOT)
700 *str++ = '!';
701
702 name = arch_cpu_feature_name(num);
703 if (!name)
704 str = strfmt("%sFEATURE 0x%X", pfx, num);
705 else
706 str = strfmt("%s%s", pfx, name);
707
708 break;
709 }
710
711 return str;
712}
713
714/*
715 * Initialize an alternative. The default alternative should be initialized
716 * with alt=NULL.
717 */
718static int disas_alt_init(struct disas_alt *dalt,
719 struct instruction *orig_insn,
720 struct alternative *alt)
721{
722 dalt->orig_insn = orig_insn;
723 dalt->alt = alt;
724 dalt->insn_idx = 0;
725 dalt->name = alt ? disas_alt_name(alt) : strdup("DEFAULT");
726 if (!dalt->name)
727 return -1;
728 dalt->width = strlen(dalt->name);
729
730 return 0;
731}
732
733static int disas_alt_add_insn(struct disas_alt *dalt, int index, char *insn_str,
734 int offset, int nops)
735{
736 int len;
737
738 if (index >= DISAS_ALT_INSN_MAX) {
739 WARN("Alternative %lx.%s has more instructions than supported",
740 DALT_ALTID(dalt), dalt->name);
741 return -1;
742 }
743
744 len = strlen(insn_str);
745 dalt->insn[index].str = insn_str;
746 dalt->insn[index].offset = offset;
747 dalt->insn[index].nops = nops;
748 if (len > dalt->width)
749 dalt->width = len;
750
751 return 0;
752}
753
754static int disas_alt_jump(struct disas_alt *dalt)
755{
756 struct instruction *orig_insn;
757 struct instruction *dest_insn;
758 char suffix[2] = { 0 };
759 char *str;
760 int nops;
761
762 orig_insn = dalt->orig_insn;
763 dest_insn = dalt->alt->insn;
764
765 if (orig_insn->type == INSN_NOP) {
766 if (orig_insn->len == 5)
767 suffix[0] = 'q';
768 str = strfmt("jmp%-3s %lx <%s+0x%lx>", suffix,
769 dest_insn->offset, dest_insn->sym->name,
770 dest_insn->offset - dest_insn->sym->offset);
771 nops = 0;
772 } else {
773 str = strfmt("nop%d", orig_insn->len);
774 nops = orig_insn->len;
775 }
776
777 if (!str)
778 return -1;
779
780 disas_alt_add_insn(dalt, 0, str, 0, nops);
781
782 return 1;
783}
784
785/*
786 * Disassemble an exception table alternative.
787 */
788static int disas_alt_extable(struct disas_alt *dalt)
789{
790 struct instruction *alt_insn;
791 char *str;
792
793 alt_insn = dalt->alt->insn;
794 str = strfmt("resume at 0x%lx <%s+0x%lx>",
795 alt_insn->offset, alt_insn->sym->name,
796 alt_insn->offset - alt_insn->sym->offset);
797 if (!str)
798 return -1;
799
800 disas_alt_add_insn(dalt, 0, str, 0, 0);
801
802 return 1;
803}
804
805/*
806 * Disassemble an alternative and store instructions in the disas_alt
807 * structure. Return the number of instructions in the alternative.
808 */
809static int disas_alt_group(struct disas_context *dctx, struct disas_alt *dalt)
810{
811 struct objtool_file *file;
812 struct instruction *insn;
813 int offset;
814 char *str;
815 int count;
816 int nops;
817 int err;
818
819 file = dctx->file;
820 count = 0;
821 offset = 0;
822 nops = 0;
823
824 alt_for_each_insn(file, DALT_GROUP(dalt), insn) {
825
826 disas_insn_alt(dctx, insn);
827 str = strdup(disas_result(dctx));
828 if (!str)
829 return -1;
830
831 nops = insn->type == INSN_NOP ? insn->len : 0;
832 err = disas_alt_add_insn(dalt, count, str, offset, nops);
833 if (err)
834 break;
835 offset += insn->len;
836 count++;
837 }
838
839 return count;
840}
841
842/*
843 * Disassemble the default alternative.
844 */
845static int disas_alt_default(struct disas_context *dctx, struct disas_alt *dalt)
846{
847 char *str;
848 int nops;
849 int err;
850
851 if (DALT_GROUP(dalt))
852 return disas_alt_group(dctx, dalt);
853
854 /*
855 * Default alternative with no alt_group: this is the default
856 * code associated with either a jump table or an exception
857 * table and no other instruction alternatives. In that case
858 * the default alternative is made of a single instruction.
859 */
860 disas_insn(dctx, dalt->orig_insn);
861 str = strdup(disas_result(dctx));
862 if (!str)
863 return -1;
864 nops = dalt->orig_insn->type == INSN_NOP ? dalt->orig_insn->len : 0;
865 err = disas_alt_add_insn(dalt, 0, str, 0, nops);
866 if (err)
867 return -1;
868
869 return 1;
870}
871
872/*
873 * For each alternative, if there is an instruction at the specified
874 * offset then print this instruction, otherwise print a blank entry.
875 * The offset is an offset from the start of the alternative.
876 *
877 * Return the offset for the next instructions to print, or -1 if all
878 * instructions have been printed.
879 */
880static int disas_alt_print_insn(struct disas_alt *dalts, int alt_count,
881 int insn_count, int offset)
882{
883 struct disas_alt *dalt;
884 int offset_next;
885 char *str;
886 int i, j;
887
888 offset_next = -1;
889
890 for (i = 0; i < alt_count; i++) {
891 dalt = &dalts[i];
892 j = dalt->insn_idx;
893 if (j == -1) {
894 printf("| %-*s ", dalt->width, "");
895 continue;
896 }
897
898 if (dalt->insn[j].offset == offset) {
899 str = dalt->insn[j].str;
900 printf("| %-*s ", dalt->width, str ?: "");
901 if (++j < insn_count) {
902 dalt->insn_idx = j;
903 } else {
904 dalt->insn_idx = -1;
905 continue;
906 }
907 } else {
908 printf("| %-*s ", dalt->width, "");
909 }
910
911 if (dalt->insn[j].offset > 0 &&
912 (offset_next == -1 ||
913 (dalt->insn[j].offset < offset_next)))
914 offset_next = dalt->insn[j].offset;
915 }
916 printf("\n");
917
918 return offset_next;
919}
920
921/*
922 * Print all alternatives side-by-side.
923 */
924static void disas_alt_print_wide(char *alt_name, struct disas_alt *dalts, int alt_count,
925 int insn_count)
926{
927 struct instruction *orig_insn;
928 int offset_next;
929 int offset;
930 int i;
931
932 orig_insn = dalts[0].orig_insn;
933
934 /*
935 * Print an header with the name of each alternative.
936 */
937 disas_print_info(stdout, orig_insn, -2, NULL);
938
939 if (strlen(alt_name) > dalts[0].width)
940 dalts[0].width = strlen(alt_name);
941 printf("| %-*s ", dalts[0].width, alt_name);
942
943 for (i = 1; i < alt_count; i++)
944 printf("| %-*s ", dalts[i].width, dalts[i].name);
945
946 printf("\n");
947
948 /*
949 * Print instructions for each alternative.
950 */
951 offset_next = 0;
952 do {
953 offset = offset_next;
954 disas_print(stdout, orig_insn->sec, orig_insn->offset + offset,
955 -2, NULL);
956 offset_next = disas_alt_print_insn(dalts, alt_count, insn_count,
957 offset);
958 } while (offset_next > offset);
959}
960
961/*
962 * Print all alternatives one above the other.
963 */
964static void disas_alt_print_compact(char *alt_name, struct disas_alt *dalts,
965 int alt_count, int insn_count)
966{
967 struct instruction *orig_insn;
968 int width;
969 int i, j;
970 int len;
971
972 orig_insn = dalts[0].orig_insn;
973
974 len = disas_print(stdout, orig_insn->sec, orig_insn->offset, 0, NULL);
975 printf("%s\n", alt_name);
976
977 /*
978 * If all alternatives have a single instruction then print each
979 * alternative on a single line. Otherwise, print alternatives
980 * one above the other with a clear separation.
981 */
982
983 if (insn_count == 1) {
984 width = 0;
985 for (i = 0; i < alt_count; i++) {
986 if (dalts[i].width > width)
987 width = dalts[i].width;
988 }
989
990 for (i = 0; i < alt_count; i++) {
991 printf("%*s= %-*s (if %s)\n", len, "", width,
992 dalts[i].insn[0].str, dalts[i].name);
993 }
994
995 return;
996 }
997
998 for (i = 0; i < alt_count; i++) {
999 printf("%*s= %s\n", len, "", dalts[i].name);
1000 for (j = 0; j < insn_count; j++) {
1001 if (!dalts[i].insn[j].str)
1002 break;
1003 disas_print(stdout, orig_insn->sec,
1004 orig_insn->offset + dalts[i].insn[j].offset, 0,
1005 "| %s\n", dalts[i].insn[j].str);
1006 }
1007 printf("%*s|\n", len, "");
1008 }
1009}
1010
1011/*
1012 * Trim NOPs in alternatives. This replaces trailing NOPs in alternatives
1013 * with a single indication of the number of bytes covered with NOPs.
1014 *
1015 * Return the maximum numbers of instructions in all alternatives after
1016 * trailing NOPs have been trimmed.
1017 */
1018static int disas_alt_trim_nops(struct disas_alt *dalts, int alt_count,
1019 int insn_count)
1020{
1021 struct disas_alt *dalt;
1022 int nops_count;
1023 const char *s;
1024 int offset;
1025 int count;
1026 int nops;
1027 int i, j;
1028
1029 count = 0;
1030 for (i = 0; i < alt_count; i++) {
1031 offset = 0;
1032 nops = 0;
1033 nops_count = 0;
1034 dalt = &dalts[i];
1035 for (j = insn_count - 1; j >= 0; j--) {
1036 if (!dalt->insn[j].str || !dalt->insn[j].nops)
1037 break;
1038 offset = dalt->insn[j].offset;
1039 free(dalt->insn[j].str);
1040 dalt->insn[j].offset = 0;
1041 dalt->insn[j].str = NULL;
1042 nops += dalt->insn[j].nops;
1043 nops_count++;
1044 }
1045
1046 /*
1047 * All trailing NOPs have been removed. If there was a single
1048 * NOP instruction then re-add it. If there was a block of
1049 * NOPs then indicate the number of bytes than the block
1050 * covers (nop*<number-of-bytes>).
1051 */
1052 if (nops_count) {
1053 s = nops_count == 1 ? "" : "*";
1054 dalt->insn[j + 1].str = strfmt("nop%s%d", s, nops);
1055 dalt->insn[j + 1].offset = offset;
1056 dalt->insn[j + 1].nops = nops;
1057 j++;
1058 }
1059
1060 if (j > count)
1061 count = j;
1062 }
1063
1064 return count + 1;
1065}
1066
1067/*
1068 * Disassemble an alternative.
1069 *
1070 * Return the last instruction in the default alternative so that
1071 * disassembly can continue with the next instruction. Return NULL
1072 * on error.
1073 */
1074static void *disas_alt(struct disas_context *dctx,
1075 struct instruction *orig_insn)
1076{
1077 struct disas_alt dalts[DISAS_ALT_MAX] = { 0 };
1078 struct instruction *last_insn = NULL;
1079 struct alternative *alt;
1080 struct disas_alt *dalt;
1081 int insn_count = 0;
1082 int alt_count = 0;
1083 char *alt_name;
1084 int count;
1085 int i, j;
1086 int err;
1087
1088 alt_name = strfmt("<%s.%lx>", disas_alt_type_name(orig_insn),
1089 orig_insn->offset);
1090 if (!alt_name) {
1091 WARN("Failed to define name for alternative at instruction 0x%lx",
1092 orig_insn->offset);
1093 goto done;
1094 }
1095
1096 /*
1097 * Initialize and disassemble the default alternative.
1098 */
1099 err = disas_alt_init(&dalts[0], orig_insn, NULL);
1100 if (err) {
1101 WARN("%s: failed to initialize default alternative", alt_name);
1102 goto done;
1103 }
1104
1105 insn_count = disas_alt_default(dctx, &dalts[0]);
1106 if (insn_count < 0) {
1107 WARN("%s: failed to disassemble default alternative", alt_name);
1108 goto done;
1109 }
1110
1111 /*
1112 * Initialize and disassemble all other alternatives.
1113 */
1114 i = 1;
1115 for (alt = orig_insn->alts; alt; alt = alt->next) {
1116 if (i >= DISAS_ALT_MAX) {
1117 WARN("%s has more alternatives than supported", alt_name);
1118 break;
1119 }
1120
1121 dalt = &dalts[i];
1122 err = disas_alt_init(dalt, orig_insn, alt);
1123 if (err) {
1124 WARN("%s: failed to disassemble alternative", alt_name);
1125 goto done;
1126 }
1127
1128 count = -1;
1129 switch (dalt->alt->type) {
1130 case ALT_TYPE_INSTRUCTIONS:
1131 count = disas_alt_group(dctx, dalt);
1132 break;
1133 case ALT_TYPE_EX_TABLE:
1134 count = disas_alt_extable(dalt);
1135 break;
1136 case ALT_TYPE_JUMP_TABLE:
1137 count = disas_alt_jump(dalt);
1138 break;
1139 }
1140 if (count < 0) {
1141 WARN("%s: failed to disassemble alternative %s",
1142 alt_name, dalt->name);
1143 goto done;
1144 }
1145
1146 insn_count = count > insn_count ? count : insn_count;
1147 i++;
1148 }
1149 alt_count = i;
1150
1151 /*
1152 * Print default and non-default alternatives.
1153 */
1154
1155 insn_count = disas_alt_trim_nops(dalts, alt_count, insn_count);
1156
1157 if (opts.wide)
1158 disas_alt_print_wide(alt_name, dalts, alt_count, insn_count);
1159 else
1160 disas_alt_print_compact(alt_name, dalts, alt_count, insn_count);
1161
1162 last_insn = orig_insn->alt_group ? orig_insn->alt_group->last_insn :
1163 orig_insn;
1164
1165done:
1166 for (i = 0; i < alt_count; i++) {
1167 free(dalts[i].name);
1168 for (j = 0; j < insn_count; j++)
1169 free(dalts[i].insn[j].str);
1170 }
1171
1172 free(alt_name);
1173
1174 return last_insn;
1175}
1176
1177/*
1178 * Disassemble a function.
1179 */
1180static void disas_func(struct disas_context *dctx, struct symbol *func)
1181{
1182 struct instruction *insn_start;
1183 struct instruction *insn;
1184
1185 printf("%s:\n", func->name);
1186 sym_for_each_insn(dctx->file, func, insn) {
1187 if (insn->alts) {
1188 insn_start = insn;
1189 insn = disas_alt(dctx, insn);
1190 if (insn)
1191 continue;
1192 /*
1193 * There was an error with disassembling
1194 * the alternative. Resume disassembling
1195 * at the current instruction, this will
1196 * disassemble the default alternative
1197 * only and continue with the code after
1198 * the alternative.
1199 */
1200 insn = insn_start;
1201 }
1202
1203 DISAS_PRINSN(dctx, insn, 0);
1204 }
1205 printf("\n");
1206}
1207
1208/*
1209 * Disassemble all warned functions.
1210 */
1211void disas_warned_funcs(struct disas_context *dctx)
1212{
1213 struct symbol *sym;
1214
1215 if (!dctx)
1216 return;
1217
1218 for_each_sym(dctx->file->elf, sym) {
1219 if (sym->warned)
1220 disas_func(dctx, sym);
1221 }
1222}
1223
1224void disas_funcs(struct disas_context *dctx)
1225{
1226 bool disas_all = !strcmp(opts.disas, "*");
1227 struct section *sec;
1228 struct symbol *sym;
1229
1230 for_each_sec(dctx->file->elf, sec) {
1231
1232 if (!(sec->sh.sh_flags & SHF_EXECINSTR))
1233 continue;
1234
1235 sec_for_each_sym(sec, sym) {
1236 /*
1237 * If the function had a warning and the verbose
1238 * option is used then the function was already
1239 * disassemble.
1240 */
1241 if (opts.verbose && sym->warned)
1242 continue;
1243
1244 if (disas_all || fnmatch(opts.disas, sym->name, 0) == 0)
1245 disas_func(dctx, sym);
1246 }
1247 }
1248}