mips: handle memory redirect for all APIs. this fixes issue #347

This commit is contained in:
Nguyen Anh Quynh 2015-12-28 15:19:30 +08:00
parent f10d79e95f
commit f935469658
2 changed files with 27 additions and 5 deletions

View file

@ -182,7 +182,7 @@ WORD_TYPE helper_le_ld_name(CPUArchState *env, target_ulong addr, int mmu_idx,
struct uc_struct *uc = env->uc; struct uc_struct *uc = env->uc;
MemoryRegion *mr = memory_mapping(uc, addr); MemoryRegion *mr = memory_mapping(uc, addr);
// memory can be unmapped while reading or fetching // memory might be still unmapped while reading or fetching
if (mr == NULL) { if (mr == NULL) {
#if defined(SOFTMMU_CODE_ACCESS) #if defined(SOFTMMU_CODE_ACCESS)
error_code = UC_ERR_FETCH_UNMAPPED; error_code = UC_ERR_FETCH_UNMAPPED;

30
uc.c
View file

@ -343,13 +343,16 @@ static bool check_mem_area(uc_engine *uc, uint64_t address, size_t size)
UNICORN_EXPORT UNICORN_EXPORT
uc_err uc_mem_read(uc_engine *uc, uint64_t address, void *_bytes, size_t size) uc_err uc_mem_read(uc_engine *uc, uint64_t address, void *_bytes, size_t size)
{ {
size_t count = 0, len;
uint8_t *bytes = _bytes; uint8_t *bytes = _bytes;
if (uc->mem_redirect) {
address = uc->mem_redirect(address);
}
if (!check_mem_area(uc, address, size)) if (!check_mem_area(uc, address, size))
return UC_ERR_READ_UNMAPPED; return UC_ERR_READ_UNMAPPED;
size_t count = 0, len;
// memory area can overlap adjacent memory blocks // memory area can overlap adjacent memory blocks
while(count < size) { while(count < size) {
MemoryRegion *mr = memory_mapping(uc, address); MemoryRegion *mr = memory_mapping(uc, address);
@ -373,13 +376,16 @@ uc_err uc_mem_read(uc_engine *uc, uint64_t address, void *_bytes, size_t size)
UNICORN_EXPORT UNICORN_EXPORT
uc_err uc_mem_write(uc_engine *uc, uint64_t address, const void *_bytes, size_t size) uc_err uc_mem_write(uc_engine *uc, uint64_t address, const void *_bytes, size_t size)
{ {
size_t count = 0, len;
const uint8_t *bytes = _bytes; const uint8_t *bytes = _bytes;
if (uc->mem_redirect) {
address = uc->mem_redirect(address);
}
if (!check_mem_area(uc, address, size)) if (!check_mem_area(uc, address, size))
return UC_ERR_WRITE_UNMAPPED; return UC_ERR_WRITE_UNMAPPED;
size_t count = 0, len;
// memory area can overlap adjacent memory blocks // memory area can overlap adjacent memory blocks
while(count < size) { while(count < size) {
MemoryRegion *mr = memory_mapping(uc, address); MemoryRegion *mr = memory_mapping(uc, address);
@ -617,6 +623,10 @@ static uc_err mem_map(uc_engine *uc, uint64_t address, size_t size, uint32_t per
UNICORN_EXPORT UNICORN_EXPORT
uc_err uc_mem_map(uc_engine *uc, uint64_t address, size_t size, uint32_t perms) uc_err uc_mem_map(uc_engine *uc, uint64_t address, size_t size, uint32_t perms)
{ {
if (uc->mem_redirect) {
address = uc->mem_redirect(address);
}
return mem_map(uc, address, size, perms, uc->memory_map(uc, address, size, perms)); return mem_map(uc, address, size, perms, uc->memory_map(uc, address, size, perms));
} }
@ -626,6 +636,10 @@ uc_err uc_mem_map_ptr(uc_engine *uc, uint64_t address, size_t size, uint32_t per
if (ptr == NULL) if (ptr == NULL)
return UC_ERR_ARG; return UC_ERR_ARG;
if (uc->mem_redirect) {
address = uc->mem_redirect(address);
}
return mem_map(uc, address, size, UC_PROT_ALL, uc->memory_map_ptr(uc, address, size, perms, ptr)); return mem_map(uc, address, size, UC_PROT_ALL, uc->memory_map_ptr(uc, address, size, perms, ptr));
} }
@ -769,6 +783,10 @@ uc_err uc_mem_protect(struct uc_struct *uc, uint64_t address, size_t size, uint3
if ((perms & ~UC_PROT_ALL) != 0) if ((perms & ~UC_PROT_ALL) != 0)
return UC_ERR_ARG; return UC_ERR_ARG;
if (uc->mem_redirect) {
address = uc->mem_redirect(address);
}
// check that user's entire requested block is mapped // check that user's entire requested block is mapped
if (!check_mem_area(uc, address, size)) if (!check_mem_area(uc, address, size))
return UC_ERR_NOMEM; return UC_ERR_NOMEM;
@ -812,6 +830,10 @@ uc_err uc_mem_unmap(struct uc_struct *uc, uint64_t address, size_t size)
if ((size & uc->target_page_align) != 0) if ((size & uc->target_page_align) != 0)
return UC_ERR_MAP; return UC_ERR_MAP;
if (uc->mem_redirect) {
address = uc->mem_redirect(address);
}
// check that user's entire requested block is mapped // check that user's entire requested block is mapped
if (!check_mem_area(uc, address, size)) if (!check_mem_area(uc, address, size))
return UC_ERR_NOMEM; return UC_ERR_NOMEM;