// Copyright © 2024 Rot127 // SPDX-License-Identifier: BSD-3 #include "test_compare.h" #include "test_detail_m680x.h" #include #include #include TestDetailM680xIdx *test_detail_m680x_idx_new() { return cs_mem_calloc(sizeof(TestDetailM680xIdx), 1); } TestDetailM680xIdx *test_detail_m680x_idx_clone(const TestDetailM680xIdx *idx) { assert(idx); TestDetailM680xIdx *clone = test_detail_m680x_idx_new(); clone->base_reg = idx->base_reg ? strdup(idx->base_reg) : NULL; clone->offset_reg = idx->offset_reg ? strdup(idx->offset_reg) : NULL; clone->flags = idx->flags_count > 0 ? cs_mem_calloc(sizeof(char *), idx->flags_count) : NULL; clone->flags_count = idx->flags_count; for (size_t i = 0; i < clone->flags_count; ++i) { clone->flags[i] = idx->flags[i] ? strdup(idx->flags[i]) : NULL; } clone->offset = idx->offset; clone->offset_addr = idx->offset_addr; clone->offset_bits = idx->offset_bits; clone->inc_dec = idx->inc_dec; return clone; } void test_detail_m680x_idx_free(TestDetailM680xIdx *idx) { if (!idx) { return; } cs_mem_free(idx->base_reg); cs_mem_free(idx->offset_reg); for (size_t i = 0; i < idx->flags_count; ++i) { cs_mem_free(idx->flags[i]); } cs_mem_free(idx->flags); cs_mem_free(idx); } TestDetailM680x *test_detail_m680x_new() { return cs_mem_calloc(sizeof(TestDetailM680x), 1); } void test_detail_m680x_free(TestDetailM680x *detail) { if (!detail) { return; } for (size_t i = 0; i < detail->operands_count; ++i) { test_detail_m680x_op_free(detail->operands[i]); } cs_mem_free(detail->operands); for (size_t i = 0; i < detail->flags_count; ++i) { cs_mem_free(detail->flags[i]); } cs_mem_free(detail->flags); cs_mem_free(detail); } TestDetailM680x *test_detail_m680x_clone(const TestDetailM680x *detail) { TestDetailM680x *clone = test_detail_m680x_new(); clone->flags_count = detail->flags_count; if (detail->flags_count > 0) { clone->flags = cs_mem_calloc(sizeof(char *), detail->flags_count); } for (size_t i = 0; i < detail->flags_count; ++i) { clone->flags[i] = detail->flags[i] ? strdup(detail->flags[i]) : NULL; } clone->operands_count = detail->operands_count; if (detail->operands_count > 0) { clone->operands = cs_mem_calloc(sizeof(TestDetailM680xOp *), detail->operands_count); } for (size_t i = 0; i < detail->operands_count; ++i) { clone->operands[i] = test_detail_m680x_op_clone(detail->operands[i]); } return clone; } TestDetailM680xOp *test_detail_m680x_op_new() { return cs_mem_calloc(sizeof(TestDetailM680xOp), 1); } TestDetailM680xOp *test_detail_m680x_op_clone(const TestDetailM680xOp *op) { TestDetailM680xOp *clone = test_detail_m680x_op_new(); clone->type = op->type ? strdup(op->type) : NULL; clone->access = op->access ? strdup(op->access) : NULL; clone->reg = op->reg ? strdup(op->reg) : NULL; clone->idx = op->idx ? test_detail_m680x_idx_clone(op->idx) : NULL; clone->imm = op->imm; clone->rel_address = op->rel_address; clone->rel_offset = op->rel_offset; clone->ext_address = op->ext_address; clone->ext_indirect = op->ext_indirect; clone->direct_addr = op->direct_addr; clone->direct_addr_set = op->direct_addr_set; clone->const_val = op->const_val; clone->size = op->size; return clone; } void test_detail_m680x_op_free(TestDetailM680xOp *op) { if (!op) { return; } cs_mem_free(op->type); cs_mem_free(op->access); cs_mem_free(op->reg); test_detail_m680x_idx_free(op->idx); cs_mem_free(op); } bool test_expected_m680x(csh *handle, const cs_m680x *actual, const TestDetailM680x *expected) { assert(handle && actual && expected); compare_uint8_ret(actual->op_count, expected->operands_count, false); for (size_t i = 0; i < actual->op_count; ++i) { const cs_m680x_op *op = &actual->operands[i]; TestDetailM680xOp *eop = expected->operands[i]; compare_enum_ret(op->type, eop->type, false); compare_enum_ret(op->access, eop->access, false); if (eop->size > 0) { compare_uint8_ret(op->size, eop->size, false); } switch (op->type) { default: fprintf(stderr, "m680x op type %" PRId32 " not handled.\n", op->type); return false; case M680X_OP_REGISTER: compare_reg_ret(*handle, op->reg, eop->reg, false); break; case M680X_OP_IMMEDIATE: compare_int32_ret(op->imm, eop->imm, false); break; case M680X_OP_EXTENDED: compare_uint16_ret(op->ext.address, eop->ext_address, false); compare_tbool_ret(op->ext.indirect, eop->ext_indirect, false); break; case M680X_OP_DIRECT: if (eop->direct_addr_set) { compare_uint8_ret(op->direct_addr, eop->direct_addr, false); } else { assert(eop->direct_addr == 0); } break; case M680X_OP_RELATIVE: compare_uint16_ret(op->rel.address, eop->rel_address, false); compare_int16_ret(op->rel.offset, eop->rel_offset, false); break; case M680X_OP_CONSTANT: compare_uint8_ret(op->const_val, eop->const_val, false); break; case M680X_OP_INDEXED: if (!eop->idx) { break; } compare_reg_ret(*handle, op->idx.base_reg, eop->idx->base_reg, false); compare_reg_ret(*handle, op->idx.offset_reg, eop->idx->offset_reg, false); if (eop->idx->offset) { compare_int16_ret(op->idx.offset, eop->idx->offset, false); } if (eop->idx->offset_addr) { compare_uint16_ret(op->idx.offset_addr, eop->idx->offset_addr, false); } if (eop->idx->offset_bits) { compare_uint8_ret(op->idx.offset_bits, eop->idx->offset_bits, false); } if (eop->idx->inc_dec) { compare_int8_ret(op->idx.inc_dec, eop->idx->inc_dec, false); } compare_bit_flags_ret(op->idx.flags, eop->idx->flags, eop->idx->flags_count, false); break; } } return true; }