mirror of
git://git.code.sf.net/p/openocd/code
synced 2025-08-05 20:32:41 +10:00
With the old checkpatch we cannot use the correct format for the SPDX tags in the file .c, in fact the C99 comments are not allowed and we had to use the block comment. With the new checkpatch, let's switch to the correct SPDX format. Change created automatically through the command: sed -i \ 's,^/\* *\(SPDX-License-Identifier: .*[^ ]\) *\*/$,// \1,' \ $(find src/ contrib/ -name \*.c) Change-Id: I6da16506baa7af718947562505dd49606d124171 Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com> Reviewed-on: https://review.openocd.org/c/openocd/+/7153 Tested-by: jenkins
135 lines
2.5 KiB
C
135 lines
2.5 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
/***************************************************************************
|
|
* Copyright (C) 2019 by Mete Balci *
|
|
* metebalci@gmail.com *
|
|
***************************************************************************/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <helper/log.h>
|
|
#include "target.h"
|
|
#include "a64_disassembler.h"
|
|
|
|
#if HAVE_CAPSTONE
|
|
|
|
#include <capstone.h>
|
|
|
|
static void print_opcode(struct command_invocation *cmd, const cs_insn *insn)
|
|
{
|
|
uint32_t opcode = 0;
|
|
|
|
memcpy(&opcode, insn->bytes, insn->size);
|
|
|
|
if (insn->size == 4) {
|
|
|
|
uint16_t opcode_high = opcode >> 16;
|
|
|
|
opcode = opcode & 0xffff;
|
|
|
|
command_print(cmd,
|
|
"0x%08" PRIx64" %04x %04x\t%s\t%s",
|
|
insn->address,
|
|
opcode,
|
|
opcode_high,
|
|
insn->mnemonic,
|
|
insn->op_str);
|
|
|
|
} else {
|
|
|
|
command_print(
|
|
cmd,
|
|
"0x%08" PRIx64" %04x\t%s\t%s",
|
|
insn->address,
|
|
opcode,
|
|
insn->mnemonic,
|
|
insn->op_str);
|
|
|
|
}
|
|
}
|
|
|
|
int a64_disassemble(struct command_invocation *cmd, struct target *target, target_addr_t address, size_t count)
|
|
{
|
|
int ret;
|
|
int csret;
|
|
csh handle;
|
|
|
|
csret = cs_open(CS_ARCH_ARM64, CS_MODE_LITTLE_ENDIAN, &handle);
|
|
|
|
if (csret != CS_ERR_OK) {
|
|
|
|
LOG_ERROR("cs_open() failed: %s", cs_strerror(csret));
|
|
return ERROR_FAIL;
|
|
|
|
}
|
|
|
|
csret = cs_option(handle, CS_OPT_SKIPDATA, CS_OPT_ON);
|
|
|
|
if (csret != CS_ERR_OK) {
|
|
|
|
LOG_ERROR("cs_option() failed: %s", cs_strerror(csret));
|
|
cs_close(&handle);
|
|
return ERROR_FAIL;
|
|
|
|
}
|
|
|
|
cs_insn *insn = cs_malloc(handle);
|
|
|
|
if (csret != CS_ERR_OK) {
|
|
|
|
LOG_ERROR("cs_malloc() failed: %s", cs_strerror(csret));
|
|
cs_close(&handle);
|
|
return ERROR_FAIL;
|
|
|
|
}
|
|
|
|
while (count > 0) {
|
|
|
|
uint8_t buffer[4];
|
|
|
|
ret = target_read_buffer(target, address, sizeof(buffer), buffer);
|
|
|
|
if (ret != ERROR_OK) {
|
|
cs_free(insn, 1);
|
|
cs_close(&handle);
|
|
return ret;
|
|
}
|
|
|
|
size_t size = sizeof(buffer);
|
|
const uint8_t *tmp = buffer;
|
|
|
|
ret = cs_disasm_iter(handle, &tmp, &size, &address, insn);
|
|
|
|
if (!ret) {
|
|
|
|
LOG_ERROR("cs_disasm_iter() failed: %s", cs_strerror(cs_errno(handle)));
|
|
cs_free(insn, 1);
|
|
cs_close(&handle);
|
|
return ERROR_FAIL;
|
|
|
|
}
|
|
|
|
print_opcode(cmd, insn);
|
|
count--;
|
|
|
|
}
|
|
|
|
cs_free(insn, 1);
|
|
cs_close(&handle);
|
|
|
|
return ERROR_OK;
|
|
}
|
|
|
|
#else
|
|
|
|
int a64_disassemble(struct command_invocation *cmd, struct target *target, target_addr_t address, size_t count)
|
|
{
|
|
command_print(cmd, "capstone disassembly framework required");
|
|
|
|
return ERROR_FAIL;
|
|
}
|
|
|
|
#endif
|