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