xref: /linux/tools/perf/util/annotate-arch/annotate-arm.c (revision c7decec2f2d2ab0366567f9e30c0e1418cece43f)
107b972ffSIan Rogers // SPDX-License-Identifier: GPL-2.0
207b972ffSIan Rogers #include <stdlib.h>
307b972ffSIan Rogers #include <linux/compiler.h>
407b972ffSIan Rogers #include <linux/zalloc.h>
507b972ffSIan Rogers #include <errno.h>
607b972ffSIan Rogers #include <regex.h>
707b972ffSIan Rogers #include "../annotate.h"
807b972ffSIan Rogers #include "../disasm.h"
907b972ffSIan Rogers 
10*0e26ba5aSIan Rogers struct arch_arm {
11*0e26ba5aSIan Rogers 	struct arch arch;
12*0e26ba5aSIan Rogers 	regex_t call_insn;
13*0e26ba5aSIan Rogers 	regex_t jump_insn;
1407b972ffSIan Rogers };
1507b972ffSIan Rogers 
arm__associate_instruction_ops(struct arch * arch,const char * name)1607b972ffSIan Rogers static const struct ins_ops *arm__associate_instruction_ops(struct arch *arch, const char *name)
1707b972ffSIan Rogers {
18*0e26ba5aSIan Rogers 	struct arch_arm *arm = container_of(arch, struct arch_arm, arch);
1907b972ffSIan Rogers 	const struct ins_ops *ops;
2007b972ffSIan Rogers 	regmatch_t match[2];
2107b972ffSIan Rogers 
2207b972ffSIan Rogers 	if (!regexec(&arm->call_insn, name, 2, match, 0))
2307b972ffSIan Rogers 		ops = &call_ops;
2407b972ffSIan Rogers 	else if (!regexec(&arm->jump_insn, name, 2, match, 0))
2507b972ffSIan Rogers 		ops = &jump_ops;
2607b972ffSIan Rogers 	else
2707b972ffSIan Rogers 		return NULL;
2807b972ffSIan Rogers 
2907b972ffSIan Rogers 	arch__associate_ins_ops(arch, name, ops);
3007b972ffSIan Rogers 	return ops;
3107b972ffSIan Rogers }
3207b972ffSIan Rogers 
arch__new_arm(const struct e_machine_and_e_flags * id,const char * cpuid __maybe_unused)33*0e26ba5aSIan Rogers const struct arch *arch__new_arm(const struct e_machine_and_e_flags *id,
34*0e26ba5aSIan Rogers 				 const char *cpuid __maybe_unused)
3507b972ffSIan Rogers {
3607b972ffSIan Rogers 	int err;
37*0e26ba5aSIan Rogers 	struct arch_arm *arm = zalloc(sizeof(*arm));
38*0e26ba5aSIan Rogers 	struct arch *arch;
3907b972ffSIan Rogers 
4007b972ffSIan Rogers 	if (!arm)
41*0e26ba5aSIan Rogers 		return NULL;
42*0e26ba5aSIan Rogers 
43*0e26ba5aSIan Rogers 	arch = &arm->arch;
44*0e26ba5aSIan Rogers 	arch->name = "arm";
45*0e26ba5aSIan Rogers 	arch->id = *id;
46*0e26ba5aSIan Rogers 	arch->objdump.comment_char	  = ';';
47*0e26ba5aSIan Rogers 	arch->objdump.skip_functions_char = '+';
48*0e26ba5aSIan Rogers 	arch->associate_instruction_ops   = arm__associate_instruction_ops;
4907b972ffSIan Rogers 
5007b972ffSIan Rogers #define ARM_CONDS "(cc|cs|eq|ge|gt|hi|le|ls|lt|mi|ne|pl|vc|vs)"
5107b972ffSIan Rogers 	err = regcomp(&arm->call_insn, "^blx?" ARM_CONDS "?$", REG_EXTENDED);
5207b972ffSIan Rogers 	if (err)
5307b972ffSIan Rogers 		goto out_free_arm;
54*0e26ba5aSIan Rogers 
5507b972ffSIan Rogers 	err = regcomp(&arm->jump_insn, "^bx?" ARM_CONDS "?$", REG_EXTENDED);
5607b972ffSIan Rogers 	if (err)
5707b972ffSIan Rogers 		goto out_free_call;
5807b972ffSIan Rogers #undef ARM_CONDS
5907b972ffSIan Rogers 
60*0e26ba5aSIan Rogers 	return arch;
6107b972ffSIan Rogers 
6207b972ffSIan Rogers out_free_call:
6307b972ffSIan Rogers 	regfree(&arm->call_insn);
6407b972ffSIan Rogers out_free_arm:
6507b972ffSIan Rogers 	free(arm);
66*0e26ba5aSIan Rogers 	errno = SYMBOL_ANNOTATE_ERRNO__ARCH_INIT_REGEXP;
67*0e26ba5aSIan Rogers 	return NULL;
6807b972ffSIan Rogers }
69