| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | /*
 | 
					
						
							|  |  |  |     armmmu.c - Memory Management Unit emulation. | 
					
						
							|  |  |  |     ARMulator extensions for the ARM7100 family. | 
					
						
							|  |  |  |     Copyright (C) 1999  Ben Williamson | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This program is free software; you can redistribute it and/or modify | 
					
						
							|  |  |  |     it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  |     the Free Software Foundation; either version 2 of the License, or | 
					
						
							|  |  |  |     (at your option) any later version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |     but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |     GNU General Public License for more details. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |     along with this program; if not, write to the Free Software | 
					
						
							|  |  |  |     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include "armdefs.h"
 | 
					
						
							|  |  |  | /* two header for arm disassemble */ | 
					
						
							|  |  |  | //#include "skyeye_arch.h"
 | 
					
						
							|  |  |  | #include "armcpu.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern mmu_ops_t xscale_mmu_ops; | 
					
						
							|  |  |  | exception_t arm_mmu_write(short size, u32 addr, uint32_t *value); | 
					
						
							|  |  |  | exception_t arm_mmu_read(short size, u32 addr, uint32_t *value); | 
					
						
							|  |  |  | #define MMU_OPS (state->mmu.ops)
 | 
					
						
							|  |  |  | ARMword skyeye_cachetype = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | mmu_init (ARMul_State * state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	state->mmu.control = 0x70; | 
					
						
							|  |  |  | 	state->mmu.translation_table_base = 0xDEADC0DE; | 
					
						
							|  |  |  | 	state->mmu.domain_access_control = 0xDEADC0DE; | 
					
						
							|  |  |  | 	state->mmu.fault_status = 0; | 
					
						
							|  |  |  | 	state->mmu.fault_address = 0; | 
					
						
							|  |  |  | 	state->mmu.process_id = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch (state->cpu->cpu_val & state->cpu->cpu_mask) { | 
					
						
							| 
									
										
										
										
											2014-03-29 21:53:07 -04:00
										 |  |  | 	//case SA1100:
 | 
					
						
							|  |  |  | 	//case SA1110:
 | 
					
						
							|  |  |  | 	//	NOTICE_LOG(ARM11, "SKYEYE: use sa11xx mmu ops\n");
 | 
					
						
							|  |  |  | 	//	state->mmu.ops = sa_mmu_ops;
 | 
					
						
							|  |  |  | 	//	break;
 | 
					
						
							|  |  |  | 	//case PXA250:
 | 
					
						
							|  |  |  | 	//case PXA270:		//xscale
 | 
					
						
							|  |  |  | 	//	NOTICE_LOG(ARM11, "SKYEYE: use xscale mmu ops\n");
 | 
					
						
							|  |  |  | 	//	state->mmu.ops = xscale_mmu_ops;
 | 
					
						
							|  |  |  | 	//	break;
 | 
					
						
							|  |  |  | 	//case 0x41807200:	//arm720t
 | 
					
						
							|  |  |  | 	//case 0x41007700:	//arm7tdmi
 | 
					
						
							|  |  |  | 	//case 0x41007100:	//arm7100
 | 
					
						
							|  |  |  | 	//	NOTICE_LOG(ARM11,  "SKYEYE: use arm7100 mmu ops\n");
 | 
					
						
							|  |  |  | 	//	state->mmu.ops = arm7100_mmu_ops;
 | 
					
						
							|  |  |  | 	//	break;
 | 
					
						
							|  |  |  | 	//case 0x41009200:
 | 
					
						
							|  |  |  | 	//	NOTICE_LOG(ARM11, "SKYEYE: use arm920t mmu ops\n");
 | 
					
						
							|  |  |  | 	//	state->mmu.ops = arm920t_mmu_ops;
 | 
					
						
							|  |  |  | 	//	break;
 | 
					
						
							|  |  |  | 	//case 0x41069260:
 | 
					
						
							|  |  |  | 	//	NOTICE_LOG(ARM11, "SKYEYE: use arm926ejs mmu ops\n");
 | 
					
						
							|  |  |  | 	//	state->mmu.ops = arm926ejs_mmu_ops;
 | 
					
						
							|  |  |  | 	//	break;
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 	/* case 0x560f5810: */ | 
					
						
							|  |  |  | 	case 0x0007b000: | 
					
						
							| 
									
										
										
										
											2014-03-29 21:53:07 -04:00
										 |  |  | 		NOTICE_LOG(ARM11, "SKYEYE: use arm11jzf-s mmu ops\n"); | 
					
						
							|  |  |  | 		//state->mmu.ops = arm1176jzf_s_mmu_ops;
 | 
					
						
							|  |  |  | 		_dbg_assert_msg_(ARM11, false, "ImplementMe: arm1176jzf_s_mmu_ops!"); | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2014-03-29 21:53:07 -04:00
										 |  |  | 		ERROR_LOG (ARM11, | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 			 "SKYEYE: armmmu.c : mmu_init: unknown cpu_val&cpu_mask 0x%x\n", | 
					
						
							|  |  |  | 			 state->cpu->cpu_val & state->cpu->cpu_mask); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	ret = state->mmu.ops.init (state); | 
					
						
							|  |  |  | 	state->mmu_inited = (ret == 0); | 
					
						
							|  |  |  | 	/* initialize mmu_read and mmu_write for disassemble */ | 
					
						
							| 
									
										
										
										
											2014-03-29 21:53:07 -04:00
										 |  |  | 	//skyeye_config_t *config  = get_current_config();
 | 
					
						
							|  |  |  | 	//generic_arch_t *arch_instance = get_arch_instance(config->arch->arch_name);
 | 
					
						
							|  |  |  | 	//arch_instance->mmu_read = arm_mmu_read;
 | 
					
						
							|  |  |  | 	//arch_instance->mmu_write = arm_mmu_write;
 | 
					
						
							| 
									
										
										
										
											2013-09-17 23:03:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | mmu_reset (ARMul_State * state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (state->mmu_inited) | 
					
						
							|  |  |  | 		mmu_exit (state); | 
					
						
							|  |  |  | 	return mmu_init (state); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | mmu_exit (ARMul_State * state) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MMU_OPS.exit (state); | 
					
						
							|  |  |  | 	state->mmu_inited = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_read_byte (ARMul_State * state, ARMword virt_addr, ARMword * data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return MMU_OPS.read_byte (state, virt_addr, data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_read_halfword (ARMul_State * state, ARMword virt_addr, ARMword * data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return MMU_OPS.read_halfword (state, virt_addr, data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_read_word (ARMul_State * state, ARMword virt_addr, ARMword * data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return MMU_OPS.read_word (state, virt_addr, data); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_write_byte (ARMul_State * state, ARMword virt_addr, ARMword data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	fault_t fault; | 
					
						
							|  |  |  | 	//static int count = 0;
 | 
					
						
							|  |  |  | 	//count ++;
 | 
					
						
							|  |  |  | 	fault = MMU_OPS.write_byte (state, virt_addr, data); | 
					
						
							|  |  |  | 	return fault; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_write_halfword (ARMul_State * state, ARMword virt_addr, ARMword data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	fault_t fault; | 
					
						
							|  |  |  | 	//static int count = 0;
 | 
					
						
							|  |  |  | 	//count ++;
 | 
					
						
							|  |  |  | 	fault = MMU_OPS.write_halfword (state, virt_addr, data); | 
					
						
							|  |  |  | 	return fault; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_write_word (ARMul_State * state, ARMword virt_addr, ARMword data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	fault_t fault; | 
					
						
							|  |  |  | 	fault = MMU_OPS.write_word (state, virt_addr, data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*used for debug for MMU*
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	   if (!fault){ | 
					
						
							|  |  |  | 	   ARMword tmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	   if (mmu_read_word(state, virt_addr, &tmp)){ | 
					
						
							|  |  |  | 	   err_msg("load back\n"); | 
					
						
							|  |  |  | 	   exit(-1); | 
					
						
							|  |  |  | 	   }else{ | 
					
						
							|  |  |  | 	   if (tmp != data){ | 
					
						
							|  |  |  | 	   err_msg("load back not equal %d %x\n", count, virt_addr); | 
					
						
							|  |  |  | 	   } | 
					
						
							|  |  |  | 	   } | 
					
						
							|  |  |  | 	   } | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return fault; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fault_t | 
					
						
							|  |  |  | mmu_load_instr (ARMul_State * state, ARMword virt_addr, ARMword * instr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return MMU_OPS.load_instr (state, virt_addr, instr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ARMword | 
					
						
							|  |  |  | mmu_mrc (ARMul_State * state, ARMword instr, ARMword * value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return MMU_OPS.mrc (state, instr, value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | mmu_mcr (ARMul_State * state, ARMword instr, ARMword value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MMU_OPS.mcr (state, instr, value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*ywc 20050416*/ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | mmu_v2p_dbct (ARMul_State * state, ARMword virt_addr, ARMword * phys_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return (MMU_OPS.v2p_dbct (state, virt_addr, phys_addr)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-29 21:53:07 -04:00
										 |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | ///* dis_mmu_read for disassemble */
 | 
					
						
							|  |  |  | //exception_t arm_mmu_read(short size, uint32_t addr, uint32_t * value)
 | 
					
						
							|  |  |  | //{
 | 
					
						
							|  |  |  | //	ARMul_State *state;
 | 
					
						
							|  |  |  | //	ARM_CPU_State *cpu = get_current_cpu();
 | 
					
						
							|  |  |  | //	state = &cpu->core[0];
 | 
					
						
							|  |  |  | //	switch(size){
 | 
					
						
							|  |  |  | //	case 8:
 | 
					
						
							|  |  |  | //		MMU_OPS.read_byte (state, addr, value);
 | 
					
						
							|  |  |  | //		break;
 | 
					
						
							|  |  |  | //	case 16:
 | 
					
						
							|  |  |  | //	case 32:
 | 
					
						
							|  |  |  | //		break;
 | 
					
						
							|  |  |  | //	default:
 | 
					
						
							|  |  |  | //		ERROR_LOG(ARM11, "Error size %d", size);
 | 
					
						
							|  |  |  | //		break;
 | 
					
						
							|  |  |  | //	}
 | 
					
						
							|  |  |  | //	return No_exp;
 | 
					
						
							|  |  |  | //}
 | 
					
						
							|  |  |  | ///* dis_mmu_write for disassemble */
 | 
					
						
							|  |  |  | //exception_t arm_mmu_write(short size, uint32_t addr, uint32_t *value)
 | 
					
						
							|  |  |  | //{
 | 
					
						
							|  |  |  | //	ARMul_State *state;
 | 
					
						
							|  |  |  | //	ARM_CPU_State *cpu = get_current_cpu();
 | 
					
						
							|  |  |  | //		state = &cpu->core[0];
 | 
					
						
							|  |  |  | //	switch(size){
 | 
					
						
							|  |  |  | //	case 8:
 | 
					
						
							|  |  |  | //		MMU_OPS.write_byte (state, addr, value);
 | 
					
						
							|  |  |  | //		break;
 | 
					
						
							|  |  |  | //	case 16:
 | 
					
						
							|  |  |  | //	case 32:
 | 
					
						
							|  |  |  | //		break;
 | 
					
						
							|  |  |  | //	default:
 | 
					
						
							|  |  |  | //		printf("In %s error size %d Line %d\n", __func__, size, __LINE__);
 | 
					
						
							|  |  |  | //		break;
 | 
					
						
							|  |  |  | //	}
 | 
					
						
							|  |  |  | //	return No_exp;
 | 
					
						
							|  |  |  | //}
 |