Skip to content

Commit

Permalink
[NFC] target/riscv: refactor riscv_read_memory,riscv_write_memory
Browse files Browse the repository at this point in the history
Reduce duplicate code
  • Loading branch information
zqb-all committed Oct 24, 2024
1 parent dd16db8 commit ac5f95d
Showing 1 changed file with 36 additions and 52 deletions.
88 changes: 36 additions & 52 deletions src/target/riscv/riscv.c
Original file line number Diff line number Diff line change
Expand Up @@ -3068,7 +3068,7 @@ static int check_virt_memory_access(struct target *target, target_addr_t address
// TODO: This assumes that size of each page is 4 KiB, which is not necessarily the case.
const bool crosses_page_boundary = RISCV_PGBASE(address + size * count - 1) != RISCV_PGBASE(address);
if (is_misaligned && crosses_page_boundary) {
LOG_TARGET_ERROR(target, "Mis-aligned memory read|write (address=0x%" PRIx64 ", size=%d, count=%d)"
LOG_TARGET_ERROR(target, "Mis-aligned memory read|write (address=0x%" TARGET_PRIxADDR ", size=%d, count=%d)"
" would access an element across page boundary. This is not supported.",
address, size, count);
return ERROR_FAIL;
Expand All @@ -3083,51 +3083,6 @@ static int riscv_read_phys_memory(struct target *target, target_addr_t phys_addr
return r->read_memory(target, phys_address, size, count, buffer, size);
}

static int riscv_read_memory(struct target *target, target_addr_t address,
uint32_t size, uint32_t count, uint8_t *buffer)
{
if (count == 0) {
LOG_TARGET_WARNING(target, "0-length read from 0x%" TARGET_PRIxADDR, address);
return ERROR_OK;
}

int mmu_enabled;
int result = riscv_mmu(target, &mmu_enabled);
if (result != ERROR_OK)
return result;

RISCV_INFO(r);

if (!mmu_enabled)
return r->read_memory(target, address, size, count, buffer, size);

result = check_virt_memory_access(target, address, size, count);
if (result != ERROR_OK)
return result;

uint32_t current_count = 0;
while (current_count < count) {
target_addr_t physical_addr;
result = target->type->virt2phys(target, address, &physical_addr);
if (result != ERROR_OK) {
LOG_TARGET_ERROR(target, "Address translation failed.");
return result;
}

/* TODO: For simplicity, this algorithm assumes the worst case - the smallest possible page size,
* which is 4 KiB. The algorithm can be improved to detect the real page size, and allow to use larger
* memory transfers and avoid extra unnecessary virt2phys address translations. */
uint32_t chunk_count = MIN(count - current_count, (RISCV_PGSIZE - RISCV_PGOFFSET(address)) / size);
result = r->read_memory(target, physical_addr, size, chunk_count, buffer + current_count * size, size);
if (result != ERROR_OK)
return result;

current_count += chunk_count;
address += chunk_count * size;
}
return ERROR_OK;
}

static int riscv_write_phys_memory(struct target *target, target_addr_t phys_address,
uint32_t size, uint32_t count, const uint8_t *buffer)
{
Expand All @@ -3137,11 +3092,16 @@ static int riscv_write_phys_memory(struct target *target, target_addr_t phys_add
return tt->write_memory(target, phys_address, size, count, buffer);
}

static int riscv_write_memory(struct target *target, target_addr_t address,
uint32_t size, uint32_t count, const uint8_t *buffer)
static int riscv_rw_memory(struct target *target, target_addr_t address, uint32_t size,
uint32_t count, uint8_t *read_buffer, const uint8_t *write_buffer)
{
/* Exactly one of the buffers must be set, the other must be NULL */
assert(!!read_buffer != !!write_buffer);

const bool is_write = write_buffer ? true : false;
if (count == 0) {
LOG_TARGET_WARNING(target, "0-length write to 0x%" TARGET_PRIxADDR, address);
LOG_TARGET_WARNING(target, "0-length %s 0x%" TARGET_PRIxADDR,
is_write ? "write to" : "read from", address);
return ERROR_OK;
}

Expand All @@ -3150,12 +3110,18 @@ static int riscv_write_memory(struct target *target, target_addr_t address,
if (result != ERROR_OK)
return result;

RISCV_INFO(r);
struct target_type *tt = get_target_type(target);
if (!tt)
return ERROR_FAIL;

if (!mmu_enabled)
return tt->write_memory(target, address, size, count, buffer);

if (!mmu_enabled) {
if (is_write)
return tt->write_memory(target, address, size, count, write_buffer);
else
return r->read_memory(target, address, size, count, read_buffer, size);
}

result = check_virt_memory_access(target, address, size, count);
if (result != ERROR_OK)
Expand All @@ -3170,8 +3136,14 @@ static int riscv_write_memory(struct target *target, target_addr_t address,
return result;
}

/* TODO: For simplicity, this algorithm assumes the worst case - the smallest possible page size,
* which is 4 KiB. The algorithm can be improved to detect the real page size, and allow to use larger
* memory transfers and avoid extra unnecessary virt2phys address translations. */
uint32_t chunk_count = MIN(count - current_count, (RISCV_PGSIZE - RISCV_PGOFFSET(address)) / size);
result = tt->write_memory(target, physical_addr, size, chunk_count, buffer + current_count * size);
if (is_write)
result = tt->write_memory(target, physical_addr, size, chunk_count, write_buffer + current_count * size);
else
result = r->read_memory(target, physical_addr, size, chunk_count, read_buffer + current_count * size, size);
if (result != ERROR_OK)
return result;

Expand All @@ -3181,6 +3153,18 @@ static int riscv_write_memory(struct target *target, target_addr_t address,
return ERROR_OK;
}

static int riscv_read_memory(struct target *target, target_addr_t address,
uint32_t size, uint32_t count, uint8_t *buffer)
{
return riscv_rw_memory(target, address, size, count, buffer, NULL);
}

static int riscv_write_memory(struct target *target, target_addr_t address,
uint32_t size, uint32_t count, const uint8_t *buffer)
{
return riscv_rw_memory(target, address, size, count, NULL, buffer);
}

static const char *riscv_get_gdb_arch(const struct target *target)
{
switch (riscv_xlen(target)) {
Expand Down

0 comments on commit ac5f95d

Please sign in to comment.