Compare commits
	
		
			38 Commits
		
	
	
		
			391f9bb808
			...
			msvc_compa
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c8679fca85 | |||
| f0ada1ba8c | |||
| 09b01af3fa | |||
| 9c8b72693e | |||
| c409e7b7ca | |||
| 2f05083cf0 | |||
| e934049dd4 | |||
|   | 94f796ebdb | ||
| 836ba269e3 | |||
| c8681096be | |||
| adeffe47ad | |||
| d95846a849 | |||
| af887c286f | |||
| 4ddf50162c | |||
| da819d8890 | |||
| 5ef5d57d30 | |||
| d7bddd825c | |||
| 15f46a87db | |||
| fc1ae4d57d | |||
| d0f3a120fd | |||
| c592a26346 | |||
| e68918c2e8 | |||
| 473f8a5a17 | |||
| 2f4b5bd9b2 | |||
| 23b9741adf | |||
| 5d8da08ce5 | |||
| a249aea703 | |||
| e432dd8208 | |||
| 8c385647dd | |||
| aaceecd5dc | |||
| 4b3f5a6b0c | |||
| d41e1d816a | |||
| a35974c9f5 | |||
| 9c456ba8f2 | |||
| c57884caee | |||
| cf7b62a3f9 | |||
| f2bf6d682a | |||
| a1fa8877f7 | 
							
								
								
									
										161
									
								
								CMakeLists.txt
									
									
									
									
									
								
							
							
						
						
									
										161
									
								
								CMakeLists.txt
									
									
									
									
									
								
							| @@ -1,12 +1,13 @@ | |||||||
| cmake_minimum_required(VERSION 3.12) | cmake_minimum_required(VERSION 3.12) | ||||||
|  | ############################################################################### | ||||||
| project(tgc VERSION 1.0.0) | # | ||||||
|  | ############################################################################### | ||||||
|  | project(dbt-rise-tgc VERSION 1.0.0) | ||||||
|  |  | ||||||
| include(GNUInstallDirs) | include(GNUInstallDirs) | ||||||
|  |  | ||||||
| conan_basic_setup() | find_package(elfio) | ||||||
|  |  | ||||||
| find_package(Boost COMPONENTS program_options system thread filesystem REQUIRED) |  | ||||||
| if(WITH_LLVM) | if(WITH_LLVM) | ||||||
|     if(DEFINED ENV{LLVM_HOME}) |     if(DEFINED ENV{LLVM_HOME}) | ||||||
|         find_path (LLVM_DIR LLVM-Config.cmake $ENV{LLVM_HOME}/lib/cmake/llvm) |         find_path (LLVM_DIR LLVM-Config.cmake $ENV{LLVM_HOME}/lib/cmake/llvm) | ||||||
| @@ -27,19 +28,17 @@ endif() | |||||||
| add_subdirectory(softfloat) | add_subdirectory(softfloat) | ||||||
|  |  | ||||||
| # library files | # library files | ||||||
| FILE(GLOB RiscVSCHeaders ${CMAKE_CURRENT_SOURCE_DIR}/incl/sysc/*.h ${CMAKE_CURRENT_SOURCE_DIR}/incl/sysc/*/*.h) | FILE(GLOB TGC_SOURCES | ||||||
| set(LIB_HEADERS tgfscVSCHeaders} ) |     ${CMAKE_CURRENT_SOURCE_DIR}/src/iss/*.cpp  | ||||||
|  |     ${CMAKE_CURRENT_SOURCE_DIR}/src/vm/interp/vm_*.cpp | ||||||
|  | ) | ||||||
| set(LIB_SOURCES  | set(LIB_SOURCES  | ||||||
|     src/vm/fp_functions.cpp |     src/vm/fp_functions.cpp | ||||||
|     src/plugin/instruction_count.cpp |     src/plugin/instruction_count.cpp | ||||||
|     src/plugin/cycle_estimate.cpp |     src/plugin/cycle_estimate.cpp | ||||||
|  |     ${TGC_SOURCES} | ||||||
| ) | ) | ||||||
| if(EXISTS src/iss/tgf_b.cpp) |  | ||||||
|     set(LIB_SOURCES ${LIB_SOURCES}  src/iss/tgf_b.cpp src/vm/interp/vm_tgf_b.cpp) |  | ||||||
| endif() |  | ||||||
| if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/src/iss/tgf_c.cpp) |  | ||||||
|     set(LIB_SOURCES ${LIB_SOURCES}  src/iss/tgf_c.cpp src/vm/interp/vm_tgf_c.cpp) |  | ||||||
| endif() |  | ||||||
| if(WITH_LLVM) | if(WITH_LLVM) | ||||||
|   set(LIB_SOURCES ${LIB_SOURCES} |   set(LIB_SOURCES ${LIB_SOURCES} | ||||||
|     src/vm/llvm/fp_impl.cpp |     src/vm/llvm/fp_impl.cpp | ||||||
| @@ -49,82 +48,122 @@ set(LIB_SOURCES ${LIB_SOURCES} | |||||||
| endif() | endif() | ||||||
|  |  | ||||||
| # Define the library | # Define the library | ||||||
| add_library(${PROJECT_NAME} SHARED ${LIB_SOURCES}) | add_library(${PROJECT_NAME} ${LIB_SOURCES}) | ||||||
| # list code gen dependencies | # list code gen dependencies | ||||||
| if(TARGET TGF_B_src) | if(TARGET ${CORE_NAME}_cpp) | ||||||
|     add_dependencies(${PROJECT_NAME} TGF_B_src) |     add_dependencies(${PROJECT_NAME} ${CORE_NAME}_cpp) | ||||||
| endif() |  | ||||||
| if(TARGET TGF_C_src) |  | ||||||
|     add_dependencies(${PROJECT_NAME} TGF_C_src) |  | ||||||
| endif() | endif() | ||||||
|  |  | ||||||
|  | if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU") | ||||||
|      target_compile_options(${PROJECT_NAME} PRIVATE -Wno-shift-count-overflow) |      target_compile_options(${PROJECT_NAME} PRIVATE -Wno-shift-count-overflow) | ||||||
|  | elseif("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC") | ||||||
|  |     target_compile_options(${PROJECT_NAME} PRIVATE /wd4293) | ||||||
|  | endif() | ||||||
| target_include_directories(${PROJECT_NAME} PUBLIC incl) | target_include_directories(${PROJECT_NAME} PUBLIC incl) | ||||||
| target_link_libraries(${PROJECT_NAME} PUBLIC softfloat scc-util jsoncpp) | target_link_libraries(${PROJECT_NAME} PUBLIC softfloat scc-util jsoncpp) | ||||||
|  | if("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU") | ||||||
|     target_link_libraries(${PROJECT_NAME} PUBLIC -Wl,--whole-archive dbt-core -Wl,--no-whole-archive) |     target_link_libraries(${PROJECT_NAME} PUBLIC -Wl,--whole-archive dbt-core -Wl,--no-whole-archive) | ||||||
| target_link_libraries(${PROJECT_NAME} PUBLIC ${Boost_LIBRARIES} ) | else() | ||||||
|  |     target_link_libraries(${PROJECT_NAME} PUBLIC dbt-core) | ||||||
|  | endif() | ||||||
|  | if(TARGET CONAN_PKG::elfio) | ||||||
|  |     target_link_libraries(${PROJECT_NAME} PUBLIC CONAN_PKG::elfio) | ||||||
|  | elseif(TARGET elfio::elfio) | ||||||
|  |     target_link_libraries(${PROJECT_NAME} PUBLIC elfio::elfio) | ||||||
|  | else() | ||||||
|  |     message(FATAL_ERROR "No elfio library found, maybe a find_package() call is missing") | ||||||
|  | endif() | ||||||
|  |  | ||||||
| set_target_properties(${PROJECT_NAME} PROPERTIES | set_target_properties(${PROJECT_NAME} PROPERTIES | ||||||
|   VERSION ${PROJECT_VERSION} |   VERSION ${PROJECT_VERSION} | ||||||
|   FRAMEWORK FALSE |   FRAMEWORK FALSE | ||||||
|   PUBLIC_HEADER "${LIB_HEADERS}" # specify the public headers |  | ||||||
| ) | ) | ||||||
|  | install(TARGETS ${PROJECT_NAME} COMPONENT ${PROJECT_NAME} | ||||||
| if(SystemC_FOUND) |   EXPORT ${PROJECT_NAME}Targets            # for downstream dependencies | ||||||
| 	add_library(${PROJECT_NAME}_sc src/sysc/core_complex.cpp) |   ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}  # static lib | ||||||
| 	target_compile_definitions(${PROJECT_NAME}_sc PUBLIC WITH_SYSTEMC)  |   RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}  # binaries | ||||||
| 	target_include_directories(${PROJECT_NAME}_sc PUBLIC ../incl ${SystemC_INCLUDE_DIRS} ${CCI_INCLUDE_DIRS}) |   LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}  # shared lib | ||||||
| 	 |   FRAMEWORK DESTINATION ${CMAKE_INSTALL_LIBDIR} # for mac | ||||||
| 	if(SCV_FOUND)    |   PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME} # headers for mac (note the different component -> different package) | ||||||
| 	    target_compile_definitions(${PROJECT_NAME}_sc PUBLIC WITH_SCV) |   INCLUDES DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}             # headers | ||||||
| 	    target_include_directories(${PROJECT_NAME}_sc PUBLIC ${SCV_INCLUDE_DIRS}) |  | ||||||
| 	endif() |  | ||||||
| 	target_link_libraries(${PROJECT_NAME}_sc PUBLIC ${PROJECT_NAME} scc) |  | ||||||
| 	if(WITH_LLVM) |  | ||||||
| 		target_link_libraries(${PROJECT_NAME}_sc PUBLIC ${llvm_libs}) |  | ||||||
| 	endif() |  | ||||||
| 	set_target_properties(${PROJECT_NAME}_sc PROPERTIES |  | ||||||
| 	  VERSION ${PROJECT_VERSION} |  | ||||||
| 	  FRAMEWORK FALSE |  | ||||||
| 	  PUBLIC_HEADER "${LIB_HEADERS}" # specify the public headers |  | ||||||
| ) | ) | ||||||
| endif() | install(DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}/incl/iss COMPONENT ${PROJECT_NAME} | ||||||
|  |         DESTINATION ${CMAKE_INSTALL_INCLUDEDIR} # target directory | ||||||
|  |         FILES_MATCHING # install only matched files | ||||||
|  |         PATTERN "*.h" # select header files | ||||||
|  |         ) | ||||||
|  | ############################################################################### | ||||||
|  | # | ||||||
|  | ############################################################################### | ||||||
|  | project(tgc-sim) | ||||||
|  | find_package(Boost COMPONENTS program_options thread REQUIRED) | ||||||
|  |  | ||||||
| project("tgc-sim") |  | ||||||
| add_executable(${PROJECT_NAME} src/main.cpp) | add_executable(${PROJECT_NAME} src/main.cpp) | ||||||
| # This sets the include directory for the reference project. This is the -I flag in gcc. | # This sets the include directory for the reference project. This is the -I flag in gcc. | ||||||
|  | target_compile_definitions(${PROJECT_NAME} PRIVATE CORE_${CORE_NAME}) | ||||||
| if(WITH_LLVM) | if(WITH_LLVM) | ||||||
|     target_compile_definitions(${PROJECT_NAME} PRIVATE WITH_LLVM) |     target_compile_definitions(${PROJECT_NAME} PRIVATE WITH_LLVM) | ||||||
|     target_link_libraries(${PROJECT_NAME} PUBLIC ${llvm_libs}) |     target_link_libraries(${PROJECT_NAME} PUBLIC ${llvm_libs}) | ||||||
| endif() | endif() | ||||||
| # Links the target exe against the libraries | # Links the target exe against the libraries | ||||||
| target_link_libraries(${PROJECT_NAME} tgc) | target_link_libraries(${PROJECT_NAME} PUBLIC dbt-rise-tgc) | ||||||
| target_link_libraries(${PROJECT_NAME} jsoncpp) | if(TARGET Boost::program_options) | ||||||
| target_link_libraries(${PROJECT_NAME} ${Boost_LIBRARIES} ) |     target_link_libraries(${PROJECT_NAME} PUBLIC Boost::program_options Boost::thread) | ||||||
|  | else() | ||||||
|  |     target_link_libraries(${PROJECT_NAME} PUBLIC ${BOOST_program_options_LIBRARY} ${BOOST_thread_LIBRARY}) | ||||||
|  | endif() | ||||||
|  | target_link_libraries(${PROJECT_NAME} PUBLIC ${CMAKE_DL_LIBS}) | ||||||
| if (Tcmalloc_FOUND) | if (Tcmalloc_FOUND) | ||||||
|     target_link_libraries(${PROJECT_NAME} ${Tcmalloc_LIBRARIES}) |     target_link_libraries(${PROJECT_NAME} PUBLIC ${Tcmalloc_LIBRARIES}) | ||||||
| endif(Tcmalloc_FOUND) | endif(Tcmalloc_FOUND) | ||||||
|  |  | ||||||
| install(TARGETS tgc tgc-sim | install(TARGETS tgc-sim | ||||||
|   EXPORT ${PROJECT_NAME}Targets            # for downstream dependencies |   EXPORT ${PROJECT_NAME}Targets            # for downstream dependencies | ||||||
|   ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT libs   # static lib |   ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}  # static lib | ||||||
|   RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR} COMPONENT libs   # binaries |   RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}  # binaries | ||||||
|   LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT libs   # shared lib |   LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}  # shared lib | ||||||
|   FRAMEWORK DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT libs # for mac |   FRAMEWORK DESTINATION ${CMAKE_INSTALL_LIBDIR} # for mac | ||||||
|   PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME} COMPONENT devel   # headers for mac (note the different component -> different package) |   PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/${PROJECT_NAME}  # headers for mac (note the different component -> different package) | ||||||
|   INCLUDES DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}             # headers |   INCLUDES DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}             # headers | ||||||
| ) | ) | ||||||
|  | ############################################################################### | ||||||
|  |  | ||||||
|  |  | ||||||
| # | # | ||||||
| # SYSTEM PACKAGING (RPM, TGZ, ...) | ############################################################################### | ||||||
| # _____________________________________________________________________________ | project(dbt-rise-tgc_sc VERSION 1.0.0) | ||||||
|  |  | ||||||
| #include(CPackConfig) | include(SystemCPackage) | ||||||
|  | if(SystemC_FOUND) | ||||||
|  |     add_library(${PROJECT_NAME} src/sysc/core_complex.cpp) | ||||||
|  |     target_compile_definitions(${PROJECT_NAME} PUBLIC WITH_SYSTEMC) | ||||||
|  |     target_compile_definitions(${PROJECT_NAME} PRIVATE CORE_${CORE_NAME}) | ||||||
|  |     if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/incl/iss/arch/tgc_b.h) | ||||||
|  |         target_compile_definitions(${PROJECT_NAME} PRIVATE CORE_TGC_B) | ||||||
|  |     endif() | ||||||
|  |     if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/incl/iss/arch/tgc_c.h) | ||||||
|  |         target_compile_definitions(${PROJECT_NAME} PRIVATE CORE_TGC_C) | ||||||
|  |     endif() | ||||||
|  |     if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/incl/iss/arch/tgc_d.h) | ||||||
|  |         target_compile_definitions(${PROJECT_NAME} PRIVATE CORE_TGC_D) | ||||||
|  |     endif() | ||||||
|  |     target_link_libraries(${PROJECT_NAME} PUBLIC dbt-rise-tgc scc) | ||||||
|  |     if(WITH_LLVM) | ||||||
|  |         target_link_libraries(${PROJECT_NAME} PUBLIC ${llvm_libs}) | ||||||
|  |     endif() | ||||||
|      |      | ||||||
| # | 	set(LIB_HEADERS ${CMAKE_CURRENT_SOURCE_DIR}/incl/sysc/core_complex.h) | ||||||
| # CMAKE PACKAGING (for other CMake projects to use this one easily) |     set_target_properties(${PROJECT_NAME} PROPERTIES | ||||||
| # _____________________________________________________________________________ |       VERSION ${PROJECT_VERSION} | ||||||
|  |       FRAMEWORK FALSE | ||||||
|  |       PUBLIC_HEADER "${LIB_HEADERS}" # specify the public headers | ||||||
|  |     ) | ||||||
|  |     install(TARGETS ${PROJECT_NAME} COMPONENT ${PROJECT_NAME} | ||||||
|  | 	  EXPORT ${PROJECT_NAME}Targets            # for downstream dependencies | ||||||
|  | 	  ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR}  # static lib | ||||||
|  | 	  RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}  # binaries | ||||||
|  | 	  LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}  # shared lib | ||||||
|  | 	  FRAMEWORK DESTINATION ${CMAKE_INSTALL_LIBDIR} # for mac | ||||||
|  | 	  PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}/sysc   # headers for mac (note the different component -> different package) | ||||||
|  | 	  INCLUDES DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}             # headers | ||||||
|  | 	)     | ||||||
|  | endif() | ||||||
|  |  | ||||||
| #include(PackageConfigurator) |  | ||||||
							
								
								
									
										30
									
								
								contrib/build.tcl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								contrib/build.tcl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | namespace eval Specification { | ||||||
|  |     proc buildproc { args } { | ||||||
|  |         global env | ||||||
|  |         variable installDir | ||||||
|  |         variable compiler | ||||||
|  |         variable compiler [::scsh::get_backend_compiler] | ||||||
|  |         #  set target $machine | ||||||
|  |         set target [::scsh::machine] | ||||||
|  |         set linkerOptions "" | ||||||
|  |         set preprocessorOptions "" | ||||||
|  |         set libversion $compiler | ||||||
|  |         switch -exact -- $target { | ||||||
|  |             "linux" { | ||||||
|  |             	set install_dir $::env(TGFS_INSTALL_ROOT) | ||||||
|  |                 set incldir "${install_dir}/include" | ||||||
|  |                 set libdir "${install_dir}/lib64" | ||||||
|  |                 set preprocessorOptions [concat $preprocessorOptions "-I${incldir}"] | ||||||
|  |                 # Set the Linker paths. | ||||||
|  |                 set linkerOptions [concat $linkerOptions "-Wl,-rpath,${libdir} -L${libdir} -ldbt-rise-tgc_sc"] | ||||||
|  |             } | ||||||
|  |             default { | ||||||
|  |                puts stderr "ERROR: \"$target\" is not supported, [::scsh::version]" | ||||||
|  |                return | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         ::scsh::cwr_append_ipsimbld_opts preprocessor "$preprocessorOptions" | ||||||
|  |         ::scsh::cwr_append_ipsimbld_opts linker       "$linkerOptions" | ||||||
|  |     } | ||||||
|  |     ::scsh::add_build_callback [namespace current]::buildproc | ||||||
|  | } | ||||||
							
								
								
									
										4
									
								
								contrib/tgc_import.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								contrib/tgc_import.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  |  | ||||||
|  | #include "sysc/core_complex.h" | ||||||
|  |  | ||||||
|  | void modules() { sysc::tgfs::core_complex i_core_complex("core_complex"); } | ||||||
							
								
								
									
										50
									
								
								contrib/tgc_import.tcl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								contrib/tgc_import.tcl
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | ############################################################################# | ||||||
|  | # | ||||||
|  | ############################################################################# | ||||||
|  | proc getScriptDirectory {} { | ||||||
|  |     set dispScriptFile [file normalize [info script]] | ||||||
|  |     set scriptFolder [file dirname $dispScriptFile] | ||||||
|  |     return $scriptFolder | ||||||
|  | } | ||||||
|  | if { $::env(SNPS_VP_PRODUCT) == "PAULTRA" } { | ||||||
|  |     set hardware /HARDWARE/HW/HW | ||||||
|  | } else { | ||||||
|  |     set hardware /HARDWARE | ||||||
|  | } | ||||||
|  |  | ||||||
|  | set scriptDir [getScriptDirectory] | ||||||
|  | set top_design_name core_complex | ||||||
|  | set clocks clk_i | ||||||
|  | set resets rst_i | ||||||
|  | set model_prefix "i_" | ||||||
|  | set model_postfix "" | ||||||
|  |  | ||||||
|  | ::pct::new_project | ||||||
|  | ::pct::open_library TLM2_PL | ||||||
|  | ::pct::clear_systemc_defines | ||||||
|  | ::pct::clear_systemc_include_path | ||||||
|  | ::pct::add_to_systemc_include_path $::env(TGFS_INSTALL_ROOT)/include | ||||||
|  | ::pct::set_import_protocol_generation_flag false | ||||||
|  | ::pct::set_update_existing_encaps_flag true | ||||||
|  | ::pct::set_dynamic_port_arrays_flag true | ||||||
|  | ::pct::set_import_scml_properties_flag true | ||||||
|  | ::pct::load_modules --set-category modules tgc_import.cc | ||||||
|  |  | ||||||
|  | # Set Port Protocols correctly | ||||||
|  | set block ${top_design_name} | ||||||
|  | foreach clock ${clocks} { | ||||||
|  | 	::pct::set_block_port_protocol --set-category SYSTEM_LIBRARY:$block/${clock} SYSTEM_LIBRARY:CLOCK | ||||||
|  | } | ||||||
|  | foreach reset ${resets} { | ||||||
|  |     ::pct::set_block_port_protocol --set-category SYSTEM_LIBRARY:$block/${reset} SYSTEM_LIBRARY:RESET | ||||||
|  | } | ||||||
|  | ::pct::set_encap_port_array_size SYSTEM_LIBRARY:$block/local_irq_i 16 | ||||||
|  |  | ||||||
|  | # Set compile settings and look | ||||||
|  | set block SYSTEM_LIBRARY:${top_design_name} | ||||||
|  | ::pct::set_encap_build_script $block/${top_design_name} $scriptDir/build.tcl | ||||||
|  | ::pct::set_background_color_rgb $block 255 255 255 255 | ||||||
|  | ::pct::create_instance SYSTEM_LIBRARY:${top_design_name}  ${hardware} ${model_prefix}${top_design_name}${model_postfix} ${top_design_name}  | ||||||
|  |  | ||||||
|  | # export the result as component | ||||||
|  | ::pct::export_system_library ${top_design_name}  ${top_design_name}.xml | ||||||
 Submodule gen_input/CoreDSL-Instruction-Set-Description updated: a5f12b0659...8d9a0fb149
									
								
							| @@ -2,7 +2,7 @@ import "CoreDSL-Instruction-Set-Description/RV32I.core_desc" | |||||||
| import "CoreDSL-Instruction-Set-Description/RVM.core_desc" | import "CoreDSL-Instruction-Set-Description/RVM.core_desc" | ||||||
| import "CoreDSL-Instruction-Set-Description/RVC.core_desc" | import "CoreDSL-Instruction-Set-Description/RVC.core_desc" | ||||||
|  |  | ||||||
| Core TGF_B provides RV32I { | Core TGC_B provides RV32I { | ||||||
| 	architectural_state { | 	architectural_state { | ||||||
|         unsigned XLEN=32; |         unsigned XLEN=32; | ||||||
|         unsigned PCLEN=32; |         unsigned PCLEN=32; | ||||||
| @@ -14,7 +14,7 @@ Core TGF_B provides RV32I { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| Core TGF_C provides RV32I, RV32M, RV32IC { | Core TGC_C provides RV32I, RV32M, RV32IC { | ||||||
|     architectural_state { |     architectural_state { | ||||||
|         unsigned XLEN=32; |         unsigned XLEN=32; | ||||||
|         unsigned PCLEN=32; |         unsigned PCLEN=32; | ||||||
| @@ -25,3 +25,13 @@ Core TGF_C provides RV32I, RV32M, RV32IC { | |||||||
|         unsigned PGMASK = 0xfff; //PGSIZE-1 |         unsigned PGMASK = 0xfff; //PGSIZE-1 | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | Core TGC_D provides RV32I, RV32M, RV32IC { | ||||||
|  |     architectural_state { | ||||||
|  |         unsigned XLEN=32; | ||||||
|  |         unsigned PCLEN=32; | ||||||
|  |         // definitions for the architecture wrapper | ||||||
|  |         //                    XL    ZYXWVUTSRQPONMLKJIHGFEDCBA | ||||||
|  |         unsigned MISA_VAL = 0b01000000000000000001000100000100; | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ | |||||||
| def getRegisterSizes(){ | def getRegisterSizes(){ | ||||||
| 	def regs = registers.collect{it.size} | 	def regs = registers.collect{it.size} | ||||||
| 	regs[-1]=64 // correct for NEXT_PC | 	regs[-1]=64 // correct for NEXT_PC | ||||||
| 	regs+=[32, 32, 64] // append TRAP_STATE, PENDING_TRAP, ICOUNT | 	regs+=[32, 32, 64, 64, 64] // append TRAP_STATE, PENDING_TRAP, ICOUNT, CYCLE, INSTRET | ||||||
|     return regs |     return regs | ||||||
| } | } | ||||||
| %> | %> | ||||||
|   | |||||||
| @@ -37,7 +37,7 @@ def nativeTypeSize(int size){ | |||||||
| } | } | ||||||
| def getRegisterSizes(){ | def getRegisterSizes(){ | ||||||
|     def regs = registers.collect{nativeTypeSize(it.size)} |     def regs = registers.collect{nativeTypeSize(it.size)} | ||||||
|     regs+=[32,32, 64] // append TRAP_STATE, PENDING_TRAP, ICOUNT |     regs+=[32,32, 64, 64, 64] // append TRAP_STATE, PENDING_TRAP, ICOUNT, CYCLE, INSTRET | ||||||
|     return regs |     return regs | ||||||
| } | } | ||||||
| def getRegisterOffsets(){ | def getRegisterOffsets(){ | ||||||
| @@ -94,7 +94,9 @@ template <> struct traits<${coreDef.name.toLowerCase()}> { | |||||||
|         ${registers.collect{it.name}.join(', ')}, NUM_REGS, |         ${registers.collect{it.name}.join(', ')}, NUM_REGS, | ||||||
|         TRAP_STATE=NUM_REGS, |         TRAP_STATE=NUM_REGS, | ||||||
|         PENDING_TRAP, |         PENDING_TRAP, | ||||||
|         ICOUNT |         ICOUNT, | ||||||
|  |         CYCLE, | ||||||
|  |         INSTRET | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     using reg_t = uint${addrDataWidth}_t; |     using reg_t = uint${addrDataWidth}_t; | ||||||
| @@ -175,6 +177,8 @@ protected: | |||||||
|         }}%> |         }}%> | ||||||
|         uint32_t trap_state = 0, pending_trap = 0; |         uint32_t trap_state = 0, pending_trap = 0; | ||||||
|         uint64_t icount = 0; |         uint64_t icount = 0; | ||||||
|  |         uint64_t cycle = 0; | ||||||
|  |         uint64_t instret = 0; | ||||||
|         uint32_t last_branch; |         uint32_t last_branch; | ||||||
|     } reg; |     } reg; | ||||||
| #pragma pack(pop) | #pragma pack(pop) | ||||||
|   | |||||||
| @@ -85,6 +85,7 @@ protected: | |||||||
|  |  | ||||||
|     inline const char *name(size_t index){return traits::reg_aliases.at(index);} |     inline const char *name(size_t index){return traits::reg_aliases.at(index);} | ||||||
|  |  | ||||||
|  |     compile_func decode_inst(code_word_t instr) ; | ||||||
|     virt_addr_t execute_inst(finish_cond_e cond, virt_addr_t start, uint64_t icount_limit) override; |     virt_addr_t execute_inst(finish_cond_e cond, virt_addr_t start, uint64_t icount_limit) override; | ||||||
|  |  | ||||||
|     // some compile time constants |     // some compile time constants | ||||||
| @@ -98,46 +99,13 @@ protected: | |||||||
|     std::array<compile_func, LUT_SIZE_C> lut_00, lut_01, lut_10; |     std::array<compile_func, LUT_SIZE_C> lut_00, lut_01, lut_10; | ||||||
|     std::array<compile_func, LUT_SIZE> lut_11; |     std::array<compile_func, LUT_SIZE> lut_11; | ||||||
|  |  | ||||||
|     std::array<compile_func *, 4> qlut; |     struct instruction_pattern { | ||||||
|  |         uint32_t value; | ||||||
|  |         uint32_t mask; | ||||||
|  |         compile_func opc; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|     std::array<const uint32_t, 4> lutmasks = {{EXTR_MASK16, EXTR_MASK16, EXTR_MASK16, EXTR_MASK32}}; |     std::array<std::vector<instruction_pattern>, 4> qlut; | ||||||
|  |  | ||||||
|     void expand_bit_mask(int pos, uint32_t mask, uint32_t value, uint32_t valid, uint32_t idx, compile_func lut[], |  | ||||||
|                          compile_func f) { |  | ||||||
|         if (pos < 0) { |  | ||||||
|             lut[idx] = f; |  | ||||||
|         } else { |  | ||||||
|             auto bitmask = 1UL << pos; |  | ||||||
|             if ((mask & bitmask) == 0) { |  | ||||||
|                 expand_bit_mask(pos - 1, mask, value, valid, idx, lut, f); |  | ||||||
|             } else { |  | ||||||
|                 if ((valid & bitmask) == 0) { |  | ||||||
|                     expand_bit_mask(pos - 1, mask, value, valid, (idx << 1), lut, f); |  | ||||||
|                     expand_bit_mask(pos - 1, mask, value, valid, (idx << 1) + 1, lut, f); |  | ||||||
|                 } else { |  | ||||||
|                     auto new_val = idx << 1; |  | ||||||
|                     if ((value & bitmask) != 0) new_val++; |  | ||||||
|                     expand_bit_mask(pos - 1, mask, value, valid, new_val, lut, f); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     inline uint32_t extract_fields(uint32_t val) { return extract_fields(29, val >> 2, lutmasks[val & 0x3], 0); } |  | ||||||
|  |  | ||||||
|     uint32_t extract_fields(int pos, uint32_t val, uint32_t mask, uint32_t lut_val) { |  | ||||||
|         if (pos >= 0) { |  | ||||||
|             auto bitmask = 1UL << pos; |  | ||||||
|             if ((mask & bitmask) == 0) { |  | ||||||
|                 lut_val = extract_fields(pos - 1, val, mask, lut_val); |  | ||||||
|             } else { |  | ||||||
|                 auto new_val = lut_val << 1; |  | ||||||
|                 if ((val & bitmask) != 0) new_val++; |  | ||||||
|                 lut_val = extract_fields(pos - 1, val, mask, new_val); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return lut_val; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     inline void raise(uint16_t trap_id, uint16_t cause){ |     inline void raise(uint16_t trap_id, uint16_t cause){ | ||||||
|         auto trap_val =  0x80ULL << 24 | (cause << 16) | trap_id; |         auto trap_val =  0x80ULL << 24 | (cause << 16) | trap_id; | ||||||
| @@ -147,8 +115,6 @@ protected: | |||||||
|  |  | ||||||
|     inline void leave(unsigned lvl){ |     inline void leave(unsigned lvl){ | ||||||
|         this->core.leave_trap(lvl); |         this->core.leave_trap(lvl); | ||||||
|         auto pc_val = super::template read_mem<reg_t>(traits::CSR, (lvl << 8) + 0x41); |  | ||||||
|         this->template get_reg<reg_t>(traits::NEXT_PC) = pc_val; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     inline void wait(unsigned type){ |     inline void wait(unsigned type){ | ||||||
| @@ -157,14 +123,42 @@ protected: | |||||||
|  |  | ||||||
|     template<typename T> |     template<typename T> | ||||||
|     T& pc_assign(T& val){super::ex_info.branch_taken=true; return val;} |     T& pc_assign(T& val){super::ex_info.branch_taken=true; return val;} | ||||||
|     inline uint8_t readSpace1(typename super::mem_type_e space, uint64_t addr){return super::template read_mem<uint8_t>(space, addr);} |     inline uint8_t readSpace1(typename super::mem_type_e space, uint64_t addr){ | ||||||
|     inline uint16_t readSpace2(typename super::mem_type_e space, uint64_t addr){return super::template read_mem<uint16_t>(space, addr);} |         auto ret = super::template read_mem<uint8_t>(space, addr); | ||||||
|     inline uint32_t readSpace4(typename super::mem_type_e space, uint64_t addr){return super::template read_mem<uint32_t>(space, addr);} |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|     inline uint64_t readSpace8(typename super::mem_type_e space, uint64_t addr){return super::template read_mem<uint64_t>(space, addr);} |         return ret; | ||||||
|     inline void writeSpace1(typename super::mem_type_e space, uint64_t addr, uint8_t data){super::write_mem(space, addr, data);} |     } | ||||||
|     inline void writeSpace2(typename super::mem_type_e space, uint64_t addr, uint16_t data){super::write_mem(space, addr, data);} |     inline uint16_t readSpace2(typename super::mem_type_e space, uint64_t addr){ | ||||||
|     inline void writeSpace4(typename super::mem_type_e space, uint64_t addr, uint32_t data){super::write_mem(space, addr, data);} |         auto ret = super::template read_mem<uint16_t>(space, addr); | ||||||
|     inline void writeSpace8(typename super::mem_type_e space, uint64_t addr, uint64_t data){super::write_mem(space, addr, data);} |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |         return ret; | ||||||
|  |     } | ||||||
|  |     inline uint32_t readSpace4(typename super::mem_type_e space, uint64_t addr){ | ||||||
|  |         auto ret = super::template read_mem<uint32_t>(space, addr); | ||||||
|  |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |         return ret; | ||||||
|  |     } | ||||||
|  |     inline uint64_t readSpace8(typename super::mem_type_e space, uint64_t addr){ | ||||||
|  |         auto ret = super::template read_mem<uint64_t>(space, addr); | ||||||
|  |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |         return ret; | ||||||
|  |     } | ||||||
|  |     inline void writeSpace1(typename super::mem_type_e space, uint64_t addr, uint8_t data){ | ||||||
|  |         super::write_mem(space, addr, data); | ||||||
|  |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |     } | ||||||
|  |     inline void writeSpace2(typename super::mem_type_e space, uint64_t addr, uint16_t data){ | ||||||
|  |         super::write_mem(space, addr, data); | ||||||
|  |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |     } | ||||||
|  |     inline void writeSpace4(typename super::mem_type_e space, uint64_t addr, uint32_t data){ | ||||||
|  |         super::write_mem(space, addr, data); | ||||||
|  |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |     } | ||||||
|  |     inline void writeSpace8(typename super::mem_type_e space, uint64_t addr, uint64_t data){ | ||||||
|  |         super::write_mem(space, addr, data); | ||||||
|  |         if(this->template get_reg<uint32_t>(traits::TRAP_STATE)) throw 0; | ||||||
|  |     } | ||||||
|     template<unsigned W, typename U, typename S = typename std::make_signed<U>::type> |     template<unsigned W, typename U, typename S = typename std::make_signed<U>::type> | ||||||
|     inline S sext(U from) { |     inline S sext(U from) { | ||||||
|         auto mask = (1ULL<<W) - 1; |         auto mask = (1ULL<<W) - 1; | ||||||
| @@ -193,31 +187,38 @@ private: | |||||||
|     /* instruction ${idx}: ${instr.name} */ |     /* instruction ${idx}: ${instr.name} */ | ||||||
|     compile_ret_t __${generator.functionName(instr.name)}(virt_addr_t& pc, code_word_t instr){ |     compile_ret_t __${generator.functionName(instr.name)}(virt_addr_t& pc, code_word_t instr){ | ||||||
|         // pre execution stuff |         // pre execution stuff | ||||||
|         this->do_sync(PRE_SYNC, ${idx}); |         auto* PC = reinterpret_cast<uint${addrDataWidth}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::PC]); | ||||||
|  |         auto NEXT_PC = reinterpret_cast<uint${addrDataWidth}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::NEXT_PC]); | ||||||
|  |         *PC=*NEXT_PC; | ||||||
|  |         auto* trap_state = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::TRAP_STATE]); | ||||||
|  |         *trap_state = *reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::PENDING_TRAP]); | ||||||
|  |         if(this->sync_exec && PRE_SYNC) this->do_sync(PRE_SYNC, ${idx}); | ||||||
|         <%instr.fields.eachLine{%>${it} |         <%instr.fields.eachLine{%>${it} | ||||||
|         <%}%>if(this->disass_enabled){ |         <%}%>if(this->disass_enabled){ | ||||||
|             /* generate console output when executing the command */ |             /* generate console output when executing the command */ | ||||||
|             <%instr.disass.eachLine{%>${it} |             <%instr.disass.eachLine{%>${it} | ||||||
|             <%}%> |             <%}%> | ||||||
|         } |         } | ||||||
|         // prepare execution |  | ||||||
|         uint${addrDataWidth}_t* PC = reinterpret_cast<uint${addrDataWidth}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::PC]); |  | ||||||
|         uint${addrDataWidth}_t* NEXT_PC = reinterpret_cast<uint${addrDataWidth}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::NEXT_PC]); |  | ||||||
|         // used registers<%instr.usedVariables.each{ k,v-> |         // used registers<%instr.usedVariables.each{ k,v-> | ||||||
|             if(v.isArray) {%> |             if(v.isArray) {%> | ||||||
|         uint${v.type.size}_t* ${k} = reinterpret_cast<uint${v.type.size}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::${k}0]);<% }else{ %>  |         auto* ${k} = reinterpret_cast<uint${v.type.size}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::${k}0]);<% }else{ %>  | ||||||
|         uint${v.type.size}_t* ${k} = reinterpret_cast<uint${v.type.size}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::${k}]); |         auto* ${k} = reinterpret_cast<uint${v.type.size}_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::${k}]); | ||||||
|         <%}}%>// calculate next pc value |         <%}}%>// calculate next pc value | ||||||
|         *NEXT_PC = *PC + ${instr.length/8}; |         *NEXT_PC = *PC + ${instr.length/8}; | ||||||
|         // execute instruction |         // execute instruction | ||||||
|  |         try { | ||||||
|         <%instr.behavior.eachLine{%>${it} |         <%instr.behavior.eachLine{%>${it} | ||||||
|         <%}%>// post execution stuff |         <%}%>} catch(...){} | ||||||
|  |         // post execution stuff | ||||||
|         if(this->sync_exec && POST_SYNC) this->do_sync(POST_SYNC, ${idx}); |         if(this->sync_exec && POST_SYNC) this->do_sync(POST_SYNC, ${idx}); | ||||||
|         auto* trap_state = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::TRAP_STATE]); |  | ||||||
|         // trap check |         // trap check | ||||||
|         if(*trap_state!=0){ |         if(*trap_state!=0){ | ||||||
|             super::core.enter_trap(*trap_state, pc.val); |             super::core.enter_trap(*trap_state, pc.val, instr); | ||||||
|  |         } else { | ||||||
|  |             (*reinterpret_cast<uint64_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::ICOUNT]))++; | ||||||
|  |             (*reinterpret_cast<uint64_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::INSTRET]))++; | ||||||
|         } |         } | ||||||
|  |         (*reinterpret_cast<uint64_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::CYCLE]))++; | ||||||
|         pc.val=*NEXT_PC; |         pc.val=*NEXT_PC; | ||||||
|         return pc; |         return pc; | ||||||
|     } |     } | ||||||
| @@ -230,13 +231,13 @@ private: | |||||||
|         uint32_t* PC = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::PC]); |         uint32_t* PC = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::PC]); | ||||||
|         uint32_t* NEXT_PC = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::NEXT_PC]); |         uint32_t* NEXT_PC = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::NEXT_PC]); | ||||||
|         *NEXT_PC = *PC + ((instr & 3) == 3 ? 4 : 2); |         *NEXT_PC = *PC + ((instr & 3) == 3 ? 4 : 2); | ||||||
|         raise(0,  11); |         raise(0,  2); | ||||||
|         // post execution stuff |         // post execution stuff | ||||||
|         if(this->sync_exec && POST_SYNC) this->do_sync(POST_SYNC, static_cast<unsigned>(arch::traits<ARCH>::opcode_e::MAX_OPCODE)); |         if(this->sync_exec && POST_SYNC) this->do_sync(POST_SYNC, static_cast<unsigned>(arch::traits<ARCH>::opcode_e::MAX_OPCODE)); | ||||||
|         auto* trap_state = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::TRAP_STATE]); |         auto* trap_state = reinterpret_cast<uint32_t*>(this->regs_base_ptr+arch::traits<ARCH>::reg_byte_offsets[arch::traits<ARCH>::TRAP_STATE]); | ||||||
|         // trap check |         // trap check | ||||||
|         if(*trap_state!=0){ |         if(*trap_state!=0){ | ||||||
|             super::core.enter_trap(*trap_state, pc.val); |             super::core.enter_trap(*trap_state, pc.val, instr); | ||||||
|         } |         } | ||||||
|         pc.val=*NEXT_PC; |         pc.val=*NEXT_PC; | ||||||
|         return pc; |         return pc; | ||||||
| @@ -263,26 +264,53 @@ template <typename CODE_WORD> void debug_fn(CODE_WORD insn) { | |||||||
|  |  | ||||||
| template <typename ARCH> vm_impl<ARCH>::vm_impl() { this(new ARCH()); } | template <typename ARCH> vm_impl<ARCH>::vm_impl() { this(new ARCH()); } | ||||||
|  |  | ||||||
|  | // according to | ||||||
|  | // https://stackoverflow.com/questions/8871204/count-number-of-1s-in-binary-representation | ||||||
|  | #ifdef __GCC__ | ||||||
|  | constexpr size_t bit_count(uint32_t u) { return __builtin_popcount(u); } | ||||||
|  | #elif __cplusplus < 201402L | ||||||
|  | constexpr size_t uCount(uint32_t u) { return u - ((u >> 1) & 033333333333) - ((u >> 2) & 011111111111); } | ||||||
|  | constexpr size_t bit_count(uint32_t u) { return ((uCount(u) + (uCount(u) >> 3)) & 030707070707) % 63; } | ||||||
|  | #else | ||||||
|  | constexpr size_t bit_count(uint32_t u) { | ||||||
|  |     size_t uCount = u - ((u >> 1) & 033333333333) - ((u >> 2) & 011111111111); | ||||||
|  |     return ((uCount + (uCount >> 3)) & 030707070707) % 63; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
| template <typename ARCH> | template <typename ARCH> | ||||||
| vm_impl<ARCH>::vm_impl(ARCH &core, unsigned core_id, unsigned cluster_id) | vm_impl<ARCH>::vm_impl(ARCH &core, unsigned core_id, unsigned cluster_id) | ||||||
| : vm_base<ARCH>(core, core_id, cluster_id) { | : vm_base<ARCH>(core, core_id, cluster_id) { | ||||||
|     qlut[0] = lut_00.data(); |  | ||||||
|     qlut[1] = lut_01.data(); |  | ||||||
|     qlut[2] = lut_10.data(); |  | ||||||
|     qlut[3] = lut_11.data(); |  | ||||||
|     for (auto instr : instr_descr) { |     for (auto instr : instr_descr) { | ||||||
|         auto quantrant = instr.value & 0x3; |         auto quadrant = instr.value & 0x3; | ||||||
|         expand_bit_mask(29, lutmasks[quantrant], instr.value >> 2, instr.mask >> 2, 0, qlut[quantrant], instr.op); |         qlut[quadrant].push_back(instruction_pattern{instr.value, instr.mask, instr.op}); | ||||||
|  |     } | ||||||
|  |     for(auto& lut: qlut){ | ||||||
|  |         std::sort(std::begin(lut), std::end(lut), [](instruction_pattern const& a, instruction_pattern const& b){ | ||||||
|  |             return bit_count(a.mask) > bit_count(b.mask); | ||||||
|  |         }); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| inline bool is_count_limit_enabled(finish_cond_e cond){ | inline bool is_count_limit_enabled(finish_cond_e cond){ | ||||||
|     return (cond & finish_cond_e::COUNT_LIMIT) == finish_cond_e::COUNT_LIMIT; |     return (cond & finish_cond_e::COUNT_LIMIT) == finish_cond_e::COUNT_LIMIT; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | inline bool is_jump_to_self_enabled(finish_cond_e cond){ | ||||||
|  |     return (cond & finish_cond_e::JUMP_TO_SELF) == finish_cond_e::JUMP_TO_SELF; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename ARCH> | ||||||
|  | typename vm_impl<ARCH>::compile_func vm_impl<ARCH>::decode_inst(code_word_t instr){ | ||||||
|  |     for(auto& e: qlut[instr&0x3]){ | ||||||
|  |         if(!((instr&e.mask) ^ e.value )) return e.opc; | ||||||
|  |     } | ||||||
|  |     return &this_class::illegal_intruction; | ||||||
|  | } | ||||||
|  |  | ||||||
| template <typename ARCH> | template <typename ARCH> | ||||||
| typename vm_base<ARCH>::virt_addr_t vm_impl<ARCH>::execute_inst(finish_cond_e cond, virt_addr_t start, uint64_t icount_limit){ | typename vm_base<ARCH>::virt_addr_t vm_impl<ARCH>::execute_inst(finish_cond_e cond, virt_addr_t start, uint64_t icount_limit){ | ||||||
|     // we fetch at max 4 byte, alignment is 2 |     // we fetch at max 4 byte, alignment is 2 | ||||||
|     enum {TRAP_ID=1<<16}; |  | ||||||
|     code_word_t insn = 0; |     code_word_t insn = 0; | ||||||
|     auto *const data = (uint8_t *)&insn; |     auto *const data = (uint8_t *)&insn; | ||||||
|     auto pc=start; |     auto pc=start; | ||||||
| @@ -290,18 +318,15 @@ typename vm_base<ARCH>::virt_addr_t vm_impl<ARCH>::execute_inst(finish_cond_e co | |||||||
|             !(is_count_limit_enabled(cond) && this->core.get_icount() >= icount_limit)){ |             !(is_count_limit_enabled(cond) && this->core.get_icount() >= icount_limit)){ | ||||||
|         auto res = fetch_ins(pc, data); |         auto res = fetch_ins(pc, data); | ||||||
|         if(res!=iss::Ok){ |         if(res!=iss::Ok){ | ||||||
|             auto new_pc = super::core.enter_trap(TRAP_ID, pc.val); |             this->do_sync(POST_SYNC, std::numeric_limits<unsigned>::max()); | ||||||
|             res = fetch_ins(virt_addr_t{access_type::FETCH, new_pc}, data); |             pc.val = super::core.enter_trap(std::numeric_limits<uint64_t>::max(), pc.val, 0); | ||||||
|             if(res!=iss::Ok) throw simulation_stopped(0); |         } else { | ||||||
|         } |             if (is_jump_to_self_enabled(cond) && | ||||||
|         if ((cond & finish_cond_e::JUMP_TO_SELF) == finish_cond_e::JUMP_TO_SELF && |  | ||||||
|                     (insn == 0x0000006f || (insn&0xffff)==0xa001)) throw simulation_stopped(0); // 'J 0' or 'C.J 0' |                     (insn == 0x0000006f || (insn&0xffff)==0xa001)) throw simulation_stopped(0); // 'J 0' or 'C.J 0' | ||||||
|         auto lut_val = extract_fields(insn); |             auto f = decode_inst(insn); | ||||||
|         auto f = qlut[insn & 0x3][lut_val]; |  | ||||||
|         if (!f) |  | ||||||
|             f = &this_class::illegal_intruction; |  | ||||||
|             pc = (this->*f)(pc, insn); |             pc = (this->*f)(pc, insn); | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|     return pc; |     return pc; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								incl/iss/arch/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								incl/iss/arch/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1 @@ | |||||||
| /tgf_b.h | /tgc_*.h | ||||||
|   | |||||||
							
								
								
									
										242
									
								
								incl/iss/arch/riscv_hart_common.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										242
									
								
								incl/iss/arch/riscv_hart_common.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,242 @@ | |||||||
|  | /******************************************************************************* | ||||||
|  |  * Copyright (C) 2017, 2018, 2021 MINRES Technologies GmbH | ||||||
|  |  * All rights reserved. | ||||||
|  |  * | ||||||
|  |  * Redistribution and use in source and binary forms, with or without | ||||||
|  |  * modification, are permitted provided that the following conditions are met: | ||||||
|  |  * | ||||||
|  |  * 1. Redistributions of source code must retain the above copyright notice, | ||||||
|  |  *    this list of conditions and the following disclaimer. | ||||||
|  |  * | ||||||
|  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | ||||||
|  |  *    this list of conditions and the following disclaimer in the documentation | ||||||
|  |  *    and/or other materials provided with the distribution. | ||||||
|  |  * | ||||||
|  |  * 3. Neither the name of the copyright holder nor the names of its contributors | ||||||
|  |  *    may be used to endorse or promote products derived from this software | ||||||
|  |  *    without specific prior written permission. | ||||||
|  |  * | ||||||
|  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||||
|  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||||
|  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||||
|  |  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE | ||||||
|  |  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | ||||||
|  |  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | ||||||
|  |  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||||
|  |  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||||
|  |  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | ||||||
|  |  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | ||||||
|  |  * POSSIBILITY OF SUCH DAMAGE. | ||||||
|  |  * | ||||||
|  |  * Contributors: | ||||||
|  |  *       eyck@minres.com - initial implementation | ||||||
|  |  ******************************************************************************/ | ||||||
|  |  | ||||||
|  | #ifndef _RISCV_HART_COMMON | ||||||
|  | #define _RISCV_HART_COMMON | ||||||
|  |  | ||||||
|  | #include "iss/arch_if.h" | ||||||
|  | #include <cstdint> | ||||||
|  |  | ||||||
|  | namespace iss { | ||||||
|  | namespace arch { | ||||||
|  |  | ||||||
|  | enum { tohost_dflt = 0xF0001000, fromhost_dflt = 0xF0001040 }; | ||||||
|  |  | ||||||
|  | enum riscv_csr { | ||||||
|  |     /* user-level CSR */ | ||||||
|  |     // User Trap Setup | ||||||
|  |     ustatus = 0x000, | ||||||
|  |     uie = 0x004, | ||||||
|  |     utvec = 0x005, | ||||||
|  |     // User Trap Handling | ||||||
|  |     uscratch = 0x040, | ||||||
|  |     uepc = 0x041, | ||||||
|  |     ucause = 0x042, | ||||||
|  |     utval = 0x043, | ||||||
|  |     uip = 0x044, | ||||||
|  |     // User Floating-Point CSRs | ||||||
|  |     fflags = 0x001, | ||||||
|  |     frm = 0x002, | ||||||
|  |     fcsr = 0x003, | ||||||
|  |     // User Counter/Timers | ||||||
|  |     cycle = 0xC00, | ||||||
|  |     time = 0xC01, | ||||||
|  |     instret = 0xC02, | ||||||
|  |     hpmcounter3 = 0xC03, | ||||||
|  |     hpmcounter4 = 0xC04, | ||||||
|  |     /*...*/ | ||||||
|  |     hpmcounter31 = 0xC1F, | ||||||
|  |     cycleh = 0xC80, | ||||||
|  |     timeh = 0xC81, | ||||||
|  |     instreth = 0xC82, | ||||||
|  |     hpmcounter3h = 0xC83, | ||||||
|  |     hpmcounter4h = 0xC84, | ||||||
|  |     /*...*/ | ||||||
|  |     hpmcounter31h = 0xC9F, | ||||||
|  |     /* supervisor-level CSR */ | ||||||
|  |     // Supervisor Trap Setup | ||||||
|  |     sstatus = 0x100, | ||||||
|  |     sedeleg = 0x102, | ||||||
|  |     sideleg = 0x103, | ||||||
|  |     sie = 0x104, | ||||||
|  |     stvec = 0x105, | ||||||
|  |     scounteren = 0x106, | ||||||
|  |     // Supervisor Trap Handling | ||||||
|  |     sscratch = 0x140, | ||||||
|  |     sepc = 0x141, | ||||||
|  |     scause = 0x142, | ||||||
|  |     stval = 0x143, | ||||||
|  |     sip = 0x144, | ||||||
|  |     // Supervisor Protection and Translation | ||||||
|  |     satp = 0x180, | ||||||
|  |     /* machine-level CSR */ | ||||||
|  |     // Machine Information Registers | ||||||
|  |     mvendorid = 0xF11, | ||||||
|  |     marchid = 0xF12, | ||||||
|  |     mimpid = 0xF13, | ||||||
|  |     mhartid = 0xF14, | ||||||
|  |     // Machine Trap Setup | ||||||
|  |     mstatus = 0x300, | ||||||
|  |     misa = 0x301, | ||||||
|  |     medeleg = 0x302, | ||||||
|  |     mideleg = 0x303, | ||||||
|  |     mie = 0x304, | ||||||
|  |     mtvec = 0x305, | ||||||
|  |     mcounteren = 0x306, | ||||||
|  |     mtvt = 0x307, //CLIC | ||||||
|  |     // Machine Trap Handling | ||||||
|  |     mscratch = 0x340, | ||||||
|  |     mepc = 0x341, | ||||||
|  |     mcause = 0x342, | ||||||
|  |     mtval = 0x343, | ||||||
|  |     mip = 0x344, | ||||||
|  |     mxnti = 0x345, //CLIC | ||||||
|  |     mintstatus   = 0x346, // MRW Current interrupt levels (CLIC) - addr subject to change | ||||||
|  |     mscratchcsw  = 0x348, // MRW Conditional scratch swap on priv mode change (CLIC) | ||||||
|  |     mscratchcswl = 0x349, // MRW Conditional scratch swap on level change (CLIC) | ||||||
|  |     mintthresh   = 0x350, // MRW Interrupt-level threshold (CLIC) - addr subject to change | ||||||
|  |     mclicbase    = 0x351, // MRW Base address for CLIC memory mapped registers (CLIC) - addr subject to change | ||||||
|  |     // Physical Memory Protection | ||||||
|  |     pmpcfg0 = 0x3A0, | ||||||
|  |     pmpcfg1 = 0x3A1, | ||||||
|  |     pmpcfg2 = 0x3A2, | ||||||
|  |     pmpcfg3 = 0x3A3, | ||||||
|  |     pmpaddr0 = 0x3B0, | ||||||
|  |     pmpaddr1 = 0x3B1, | ||||||
|  |     pmpaddr2 = 0x3B2, | ||||||
|  |     pmpaddr3 = 0x3B3, | ||||||
|  |     pmpaddr4 = 0x3B4, | ||||||
|  |     pmpaddr5 = 0x3B5, | ||||||
|  |     pmpaddr6 = 0x3B6, | ||||||
|  |     pmpaddr7 = 0x3B7, | ||||||
|  |     pmpaddr8 = 0x3B8, | ||||||
|  |     pmpaddr9 = 0x3B9, | ||||||
|  |     pmpaddr10 = 0x3BA, | ||||||
|  |     pmpaddr11 = 0x3BB, | ||||||
|  |     pmpaddr12 = 0x3BC, | ||||||
|  |     pmpaddr13 = 0x3BD, | ||||||
|  |     pmpaddr14 = 0x3BE, | ||||||
|  |     pmpaddr15 = 0x3BF, | ||||||
|  |     // Machine Counter/Timers | ||||||
|  |     mcycle = 0xB00, | ||||||
|  |     minstret = 0xB02, | ||||||
|  |     mhpmcounter3 = 0xB03, | ||||||
|  |     mhpmcounter4 = 0xB04, | ||||||
|  |     /*...*/ | ||||||
|  |     mhpmcounter31 = 0xB1F, | ||||||
|  |     mcycleh = 0xB80, | ||||||
|  |     minstreth = 0xB82, | ||||||
|  |     mhpmcounter3h = 0xB83, | ||||||
|  |     mhpmcounter4h = 0xB84, | ||||||
|  |     /*...*/ | ||||||
|  |     mhpmcounter31h = 0xB9F, | ||||||
|  |     // Machine Counter Setup | ||||||
|  |     mhpmevent3 = 0x323, | ||||||
|  |     mhpmevent4 = 0x324, | ||||||
|  |     /*...*/ | ||||||
|  |     mhpmevent31 = 0x33F, | ||||||
|  |     // Debug/Trace Registers (shared with Debug Mode) | ||||||
|  |     tselect = 0x7A0, | ||||||
|  |     tdata1 = 0x7A1, | ||||||
|  |     tdata2 = 0x7A2, | ||||||
|  |     tdata3 = 0x7A3, | ||||||
|  |     // Debug Mode Registers | ||||||
|  |     dcsr = 0x7B0, | ||||||
|  |     dpc = 0x7B1, | ||||||
|  |     dscratch = 0x7B2 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | enum { | ||||||
|  |     PGSHIFT = 12, | ||||||
|  |     PTE_PPN_SHIFT = 10, | ||||||
|  |     // page table entry (PTE) fields | ||||||
|  |     PTE_V = 0x001,   // Valid | ||||||
|  |     PTE_R = 0x002,   // Read | ||||||
|  |     PTE_W = 0x004,   // Write | ||||||
|  |     PTE_X = 0x008,   // Execute | ||||||
|  |     PTE_U = 0x010,   // User | ||||||
|  |     PTE_G = 0x020,   // Global | ||||||
|  |     PTE_A = 0x040,   // Accessed | ||||||
|  |     PTE_D = 0x080,   // Dirty | ||||||
|  |     PTE_SOFT = 0x300 // Reserved for Software | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | template <typename T> inline bool PTE_TABLE(T PTE) { return (((PTE) & (PTE_V | PTE_R | PTE_W | PTE_X)) == PTE_V); } | ||||||
|  |  | ||||||
|  | enum { PRIV_U = 0, PRIV_S = 1, PRIV_M = 3 }; | ||||||
|  |  | ||||||
|  | enum { | ||||||
|  |     ISA_A = 1, | ||||||
|  |     ISA_B = 1 << 1, | ||||||
|  |     ISA_C = 1 << 2, | ||||||
|  |     ISA_D = 1 << 3, | ||||||
|  |     ISA_E = 1 << 4, | ||||||
|  |     ISA_F = 1 << 5, | ||||||
|  |     ISA_G = 1 << 6, | ||||||
|  |     ISA_I = 1 << 8, | ||||||
|  |     ISA_M = 1 << 12, | ||||||
|  |     ISA_N = 1 << 13, | ||||||
|  |     ISA_Q = 1 << 16, | ||||||
|  |     ISA_S = 1 << 18, | ||||||
|  |     ISA_U = 1 << 20 | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct vm_info { | ||||||
|  |     int levels; | ||||||
|  |     int idxbits; | ||||||
|  |     int ptesize; | ||||||
|  |     uint64_t ptbase; | ||||||
|  |     bool is_active() { return levels; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | class trap_load_access_fault : public trap_access { | ||||||
|  | public: | ||||||
|  |     trap_load_access_fault(uint64_t badaddr) | ||||||
|  |     : trap_access(5 << 16, badaddr) {} | ||||||
|  | }; | ||||||
|  | class illegal_instruction_fault : public trap_access { | ||||||
|  | public: | ||||||
|  |     illegal_instruction_fault(uint64_t badaddr) | ||||||
|  |     : trap_access(2 << 16, badaddr) {} | ||||||
|  | }; | ||||||
|  | class trap_instruction_page_fault : public trap_access { | ||||||
|  | public: | ||||||
|  |     trap_instruction_page_fault(uint64_t badaddr) | ||||||
|  |     : trap_access(12 << 16, badaddr) {} | ||||||
|  | }; | ||||||
|  | class trap_load_page_fault : public trap_access { | ||||||
|  | public: | ||||||
|  |     trap_load_page_fault(uint64_t badaddr) | ||||||
|  |     : trap_access(13 << 16, badaddr) {} | ||||||
|  | }; | ||||||
|  | class trap_store_page_fault : public trap_access { | ||||||
|  | public: | ||||||
|  |     trap_store_page_fault(uint64_t badaddr) | ||||||
|  |     : trap_access(15 << 16, badaddr) {} | ||||||
|  | }; | ||||||
|  | } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| /******************************************************************************* | /******************************************************************************* | ||||||
|  * Copyright (C) 2017, 2018, MINRES Technologies GmbH |  * Copyright (C) 2021 MINRES Technologies GmbH | ||||||
|  * All rights reserved. |  * All rights reserved. | ||||||
|  * |  * | ||||||
|  * Redistribution and use in source and binary forms, with or without |  * Redistribution and use in source and binary forms, with or without | ||||||
| @@ -32,11 +32,11 @@ | |||||||
|  *       eyck@minres.com - initial implementation |  *       eyck@minres.com - initial implementation | ||||||
|  ******************************************************************************/ |  ******************************************************************************/ | ||||||
|  |  | ||||||
| #ifndef _RISCV_CORE_H_ | #ifndef _RISCV_HART_M_P_H | ||||||
| #define _RISCV_CORE_H_ | #define _RISCV_HART_M_P_H | ||||||
|  |  | ||||||
|  | #include "riscv_hart_common.h" | ||||||
| #include "iss/arch/traits.h" | #include "iss/arch/traits.h" | ||||||
| #include "iss/arch_if.h" |  | ||||||
| #include "iss/instrumentation_if.h" | #include "iss/instrumentation_if.h" | ||||||
| #include "iss/log_categories.h" | #include "iss/log_categories.h" | ||||||
| #include "iss/vm_if.h" | #include "iss/vm_if.h" | ||||||
| @@ -66,116 +66,10 @@ | |||||||
| namespace iss { | namespace iss { | ||||||
| namespace arch { | namespace arch { | ||||||
|  |  | ||||||
| enum { tohost_dflt = 0xF0001000, fromhost_dflt = 0xF0001040 }; | template <typename BASE> class riscv_hart_m_p : public BASE { | ||||||
|  | protected: | ||||||
| enum riscv_csr { |     const std::array<const char, 4> lvl = {{'U', 'S', 'H', 'M'}}; | ||||||
|     /* user-level CSR */ |     const std::array<const char *, 16> trap_str = {{"" | ||||||
|     // User Trap Setup |  | ||||||
|     ustatus = 0x000, |  | ||||||
|     uie = 0x004, |  | ||||||
|     utvec = 0x005, |  | ||||||
|     // User Trap Handling |  | ||||||
|     uscratch = 0x040, |  | ||||||
|     uepc = 0x041, |  | ||||||
|     ucause = 0x042, |  | ||||||
|     utval = 0x043, |  | ||||||
|     uip = 0x044, |  | ||||||
|     // User Floating-Point CSRs |  | ||||||
|     fflags = 0x001, |  | ||||||
|     frm = 0x002, |  | ||||||
|     fcsr = 0x003, |  | ||||||
|     // User Counter/Timers |  | ||||||
|     cycle = 0xC00, |  | ||||||
|     time = 0xC01, |  | ||||||
|     instret = 0xC02, |  | ||||||
|     hpmcounter3 = 0xC03, |  | ||||||
|     hpmcounter4 = 0xC04, |  | ||||||
|     /*...*/ |  | ||||||
|     hpmcounter31 = 0xC1F, |  | ||||||
|     cycleh = 0xC80, |  | ||||||
|     timeh = 0xC81, |  | ||||||
|     instreth = 0xC82, |  | ||||||
|     hpmcounter3h = 0xC83, |  | ||||||
|     hpmcounter4h = 0xC84, |  | ||||||
|     /*...*/ |  | ||||||
|     hpmcounter31h = 0xC9F, |  | ||||||
|     /* supervisor-level CSR */ |  | ||||||
|     // Supervisor Trap Setup |  | ||||||
|     sstatus = 0x100, |  | ||||||
|     sedeleg = 0x102, |  | ||||||
|     sideleg = 0x103, |  | ||||||
|     sie = 0x104, |  | ||||||
|     stvec = 0x105, |  | ||||||
|     scounteren = 0x106, |  | ||||||
|     // Supervisor Trap Handling |  | ||||||
|     sscratch = 0x140, |  | ||||||
|     sepc = 0x141, |  | ||||||
|     scause = 0x142, |  | ||||||
|     stval = 0x143, |  | ||||||
|     sip = 0x144, |  | ||||||
|     // Supervisor Protection and Translation |  | ||||||
|     satp = 0x180, |  | ||||||
|     /* machine-level CSR */ |  | ||||||
|     // Machine Information Registers |  | ||||||
|     mvendorid = 0xF11, |  | ||||||
|     marchid = 0xF12, |  | ||||||
|     mimpid = 0xF13, |  | ||||||
|     mhartid = 0xF14, |  | ||||||
|     // Machine Trap Setup |  | ||||||
|     mstatus = 0x300, |  | ||||||
|     misa = 0x301, |  | ||||||
|     medeleg = 0x302, |  | ||||||
|     mideleg = 0x303, |  | ||||||
|     mie = 0x304, |  | ||||||
|     mtvec = 0x305, |  | ||||||
|     mcounteren = 0x306, |  | ||||||
|     // Machine Trap Handling |  | ||||||
|     mscratch = 0x340, |  | ||||||
|     mepc = 0x341, |  | ||||||
|     mcause = 0x342, |  | ||||||
|     mtval = 0x343, |  | ||||||
|     mip = 0x344, |  | ||||||
|     // Machine Protection and Translation |  | ||||||
|     pmpcfg0 = 0x3A0, |  | ||||||
|     pmpcfg1 = 0x3A1, |  | ||||||
|     pmpcfg2 = 0x3A2, |  | ||||||
|     pmpcfg3 = 0x3A3, |  | ||||||
|     pmpaddr0 = 0x3B0, |  | ||||||
|     pmpaddr1 = 0x3B1, |  | ||||||
|     /*...*/ |  | ||||||
|     pmpaddr15 = 0x3BF, |  | ||||||
|     // Machine Counter/Timers |  | ||||||
|     mcycle = 0xB00, |  | ||||||
|     minstret = 0xB02, |  | ||||||
|     mhpmcounter3 = 0xB03, |  | ||||||
|     mhpmcounter4 = 0xB04, |  | ||||||
|     /*...*/ |  | ||||||
|     mhpmcounter31 = 0xB1F, |  | ||||||
|     mcycleh = 0xB80, |  | ||||||
|     minstreth = 0xB82, |  | ||||||
|     mhpmcounter3h = 0xB83, |  | ||||||
|     mhpmcounter4h = 0xB84, |  | ||||||
|     /*...*/ |  | ||||||
|     mhpmcounter31h = 0xB9F, |  | ||||||
|     // Machine Counter Setup |  | ||||||
|     mhpmevent3 = 0x323, |  | ||||||
|     mhpmevent4 = 0x324, |  | ||||||
|     /*...*/ |  | ||||||
|     mhpmevent31 = 0x33F, |  | ||||||
|     // Debug/Trace Registers (shared with Debug Mode) |  | ||||||
|     tselect = 0x7A0, |  | ||||||
|     tdata1 = 0x7A1, |  | ||||||
|     tdata2 = 0x7A2, |  | ||||||
|     tdata3 = 0x7A3, |  | ||||||
|     // Debug Mode Registers |  | ||||||
|     dcsr = 0x7B0, |  | ||||||
|     dpc = 0x7B1, |  | ||||||
|     dscratch = 0x7B2 |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| namespace { |  | ||||||
|  |  | ||||||
| std::array<const char *, 16> trap_str = {{"" |  | ||||||
|                                               "Instruction address misaligned", // 0 |                                               "Instruction address misaligned", // 0 | ||||||
|                                               "Instruction access fault",       // 1 |                                               "Instruction access fault",       // 1 | ||||||
|                                               "Illegal instruction",            // 2 |                                               "Illegal instruction",            // 2 | ||||||
| @@ -192,65 +86,16 @@ std::array<const char *, 16> trap_str = {{"" | |||||||
|                                               "Load page fault",                // d |                                               "Load page fault",                // d | ||||||
|                                               "Reserved",                       // e |                                               "Reserved",                       // e | ||||||
|                                               "Store/AMO page fault"}}; |                                               "Store/AMO page fault"}}; | ||||||
| std::array<const char *, 12> irq_str = { |     const std::array<const char *, 12> irq_str = { | ||||||
|         {"User software interrupt", "Supervisor software interrupt", "Reserved", "Machine software interrupt", |         {"User software interrupt", "Supervisor software interrupt", "Reserved", "Machine software interrupt", | ||||||
|          "User timer interrupt", "Supervisor timer interrupt", "Reserved", "Machine timer interrupt", |          "User timer interrupt", "Supervisor timer interrupt", "Reserved", "Machine timer interrupt", | ||||||
|          "User external interrupt", "Supervisor external interrupt", "Reserved", "Machine external interrupt"}}; |          "User external interrupt", "Supervisor external interrupt", "Reserved", "Machine external interrupt"}}; | ||||||
|  |  | ||||||
| enum { |  | ||||||
|     PGSHIFT = 12, |  | ||||||
|     PTE_PPN_SHIFT = 10, |  | ||||||
|     // page table entry (PTE) fields |  | ||||||
|     PTE_V = 0x001,   // Valid |  | ||||||
|     PTE_R = 0x002,   // Read |  | ||||||
|     PTE_W = 0x004,   // Write |  | ||||||
|     PTE_X = 0x008,   // Execute |  | ||||||
|     PTE_U = 0x010,   // User |  | ||||||
|     PTE_G = 0x020,   // Global |  | ||||||
|     PTE_A = 0x040,   // Accessed |  | ||||||
|     PTE_D = 0x080,   // Dirty |  | ||||||
|     PTE_SOFT = 0x300 // Reserved for Software |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| template <typename T> inline bool PTE_TABLE(T PTE) { return (((PTE) & (PTE_V | PTE_R | PTE_W | PTE_X)) == PTE_V); } |  | ||||||
|  |  | ||||||
| enum { PRIV_U = 0, PRIV_S = 1, PRIV_M = 3 }; |  | ||||||
|  |  | ||||||
| enum { |  | ||||||
|     ISA_A = 1, |  | ||||||
|     ISA_B = 1 << 1, |  | ||||||
|     ISA_C = 1 << 2, |  | ||||||
|     ISA_D = 1 << 3, |  | ||||||
|     ISA_E = 1 << 4, |  | ||||||
|     ISA_F = 1 << 5, |  | ||||||
|     ISA_G = 1 << 6, |  | ||||||
|     ISA_I = 1 << 8, |  | ||||||
|     ISA_M = 1 << 12, |  | ||||||
|     ISA_N = 1 << 13, |  | ||||||
|     ISA_Q = 1 << 16, |  | ||||||
|     ISA_S = 1 << 18, |  | ||||||
|     ISA_U = 1 << 20 |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class trap_load_access_fault : public trap_access { |  | ||||||
| public: | public: | ||||||
|     trap_load_access_fault(uint64_t badaddr) |     using core = BASE; | ||||||
|     : trap_access(5 << 16, badaddr) {} |  | ||||||
| }; |  | ||||||
| class illegal_instruction_fault : public trap_access { |  | ||||||
| public: |  | ||||||
|     illegal_instruction_fault(uint64_t badaddr) |  | ||||||
|     : trap_access(2 << 16, badaddr) {} |  | ||||||
| }; |  | ||||||
| } // namespace |  | ||||||
|  |  | ||||||
| template <typename BASE> class riscv_hart_m_p : public BASE { |  | ||||||
| public: |  | ||||||
|     using super = BASE; |  | ||||||
|     using this_class = riscv_hart_m_p<BASE>; |     using this_class = riscv_hart_m_p<BASE>; | ||||||
|     using phys_addr_t = typename super::phys_addr_t; |     using phys_addr_t = typename core::phys_addr_t; | ||||||
|     using reg_t = typename super::reg_t; |     using reg_t = typename core::reg_t; | ||||||
|     using addr_t = typename super::addr_t; |     using addr_t = typename core::addr_t; | ||||||
|  |  | ||||||
|     using rd_csr_f = iss::status (this_class::*)(unsigned addr, reg_t &); |     using rd_csr_f = iss::status (this_class::*)(unsigned addr, reg_t &); | ||||||
|     using wr_csr_f = iss::status (this_class::*)(unsigned addr, reg_t); |     using wr_csr_f = iss::status (this_class::*)(unsigned addr, reg_t); | ||||||
| @@ -299,23 +144,40 @@ public: | |||||||
|  |  | ||||||
|         mstatus_t mstatus; |         mstatus_t mstatus; | ||||||
|  |  | ||||||
|         static const reg_t mstatus_reset_val = 0; |         static const reg_t mstatus_reset_val = 0x1800; | ||||||
|  |  | ||||||
|         void write_mstatus(T val) { |         void write_mstatus(T val) { | ||||||
|             auto mask = get_mask(); |             auto mask = get_mask() &0xff; // MPP is hardcode as 0x3 | ||||||
|             auto new_val = (mstatus.backing.val & ~mask) | (val & mask); |             auto new_val = (mstatus.backing.val & ~mask) | (val & mask); | ||||||
|             mstatus = new_val; |             mstatus = new_val; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         T satp; |  | ||||||
|  |  | ||||||
|         static constexpr uint32_t get_mask() { |         static constexpr uint32_t get_mask() { | ||||||
|             return 0x807ff9ddUL; // 0b1000 0000 0111 1111 1111 1001 1011 1011  // only machine mode is supported |             //return 0x807ff988UL; // 0b1000 0000 0111 1111 1111 1000 1000 1000  // only machine mode is supported | ||||||
|  |             //       +-SD | ||||||
|  |             //       |        +-TSR | ||||||
|  |             //       |        |+-TW | ||||||
|  |             //       |        ||+-TVM | ||||||
|  |             //       |        |||+-MXR | ||||||
|  |             //       |        ||||+-SUM | ||||||
|  |             //       |        |||||+-MPRV | ||||||
|  |             //       |        |||||| +-XS | ||||||
|  |             //       |        |||||| | +-FS | ||||||
|  |             //       |        |||||| | | +-MPP | ||||||
|  |             //       |        |||||| | | |  +-SPP | ||||||
|  |             //       |        |||||| | | |  |+-MPIE | ||||||
|  |             //       |        ||||||/|/|/|  ||   +-MIE | ||||||
|  |             return 0b00000000000000000001100010001000; | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
|  |     using hart_state_type = hart_state<reg_t>; | ||||||
|  |  | ||||||
|     constexpr reg_t get_irq_mask() { |     constexpr reg_t get_irq_mask() { | ||||||
|         return 0b101110111011; // only machine mode is supported |         return 0b100010001000; // only machine mode is supported | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     constexpr reg_t get_pc_mask() { | ||||||
|  |         return traits<BASE>::MISA_VAL&0b0100?~1:~3; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     riscv_hart_m_p(); |     riscv_hart_m_p(); | ||||||
| @@ -330,8 +192,8 @@ public: | |||||||
|     iss::status write(const address_type type, const access_type access, const uint32_t space, |     iss::status write(const address_type type, const access_type access, const uint32_t space, | ||||||
|             const uint64_t addr, const unsigned length, const uint8_t *const data) override; |             const uint64_t addr, const unsigned length, const uint8_t *const data) override; | ||||||
|  |  | ||||||
|     virtual uint64_t enter_trap(uint64_t flags) override { return riscv_hart_m_p::enter_trap(flags, fault_data); } |     virtual uint64_t enter_trap(uint64_t flags) override { return riscv_hart_m_p::enter_trap(flags, fault_data, fault_data); } | ||||||
|     virtual uint64_t enter_trap(uint64_t flags, uint64_t addr) override; |     virtual uint64_t enter_trap(uint64_t flags, uint64_t addr, uint64_t instr) override; | ||||||
|     virtual uint64_t leave_trap(uint64_t flags) override; |     virtual uint64_t leave_trap(uint64_t flags) override; | ||||||
|  |  | ||||||
|     const reg_t& get_mhartid() const { return mhartid_reg;	} |     const reg_t& get_mhartid() const { return mhartid_reg;	} | ||||||
| @@ -387,8 +249,11 @@ protected: | |||||||
|     virtual iss::status read_csr(unsigned addr, reg_t &val); |     virtual iss::status read_csr(unsigned addr, reg_t &val); | ||||||
|     virtual iss::status write_csr(unsigned addr, reg_t val); |     virtual iss::status write_csr(unsigned addr, reg_t val); | ||||||
|  |  | ||||||
|     hart_state<reg_t> state; |     hart_state_type state; | ||||||
|     uint64_t cycle_offset; |     int64_t cycle_offset{0}; | ||||||
|  |     uint64_t mcycle_csr{0}; | ||||||
|  |     int64_t instret_offset{0}; | ||||||
|  |     uint64_t minstret_csr{0}; | ||||||
|     reg_t fault_data; |     reg_t fault_data; | ||||||
|     uint64_t tohost = tohost_dflt; |     uint64_t tohost = tohost_dflt; | ||||||
|     uint64_t fromhost = fromhost_dflt; |     uint64_t fromhost = fromhost_dflt; | ||||||
| @@ -409,15 +274,23 @@ protected: | |||||||
| private: | private: | ||||||
|     iss::status read_reg(unsigned addr, reg_t &val); |     iss::status read_reg(unsigned addr, reg_t &val); | ||||||
|     iss::status write_reg(unsigned addr, reg_t val); |     iss::status write_reg(unsigned addr, reg_t val); | ||||||
|  |     iss::status read_null(unsigned addr, reg_t &val); | ||||||
|  |     iss::status write_null(unsigned addr, reg_t val){return iss::status::Ok;} | ||||||
|     iss::status read_cycle(unsigned addr, reg_t &val); |     iss::status read_cycle(unsigned addr, reg_t &val); | ||||||
|  |     iss::status write_cycle(unsigned addr, reg_t val); | ||||||
|  |     iss::status read_instret(unsigned addr, reg_t &val); | ||||||
|  |     iss::status write_instret(unsigned addr, reg_t val); | ||||||
|  |     iss::status read_tvec(unsigned addr, reg_t &val); | ||||||
|     iss::status read_time(unsigned addr, reg_t &val); |     iss::status read_time(unsigned addr, reg_t &val); | ||||||
|     iss::status read_status(unsigned addr, reg_t &val); |     iss::status read_status(unsigned addr, reg_t &val); | ||||||
|     iss::status write_status(unsigned addr, reg_t val); |     iss::status write_status(unsigned addr, reg_t val); | ||||||
|  |     iss::status write_cause(unsigned addr, reg_t val); | ||||||
|     iss::status read_ie(unsigned addr, reg_t &val); |     iss::status read_ie(unsigned addr, reg_t &val); | ||||||
|     iss::status write_ie(unsigned addr, reg_t val); |     iss::status write_ie(unsigned addr, reg_t val); | ||||||
|     iss::status read_ip(unsigned addr, reg_t &val); |     iss::status read_ip(unsigned addr, reg_t &val); | ||||||
|     iss::status write_ip(unsigned addr, reg_t val); |     iss::status write_ip(unsigned addr, reg_t val); | ||||||
|     iss::status read_hartid(unsigned addr, reg_t &val); |     iss::status read_hartid(unsigned addr, reg_t &val); | ||||||
|  |     iss::status write_epc(unsigned addr, reg_t val); | ||||||
|  |  | ||||||
|     reg_t mhartid_reg{0x0}; |     reg_t mhartid_reg{0x0}; | ||||||
|     std::function<iss::status(phys_addr_t, unsigned, uint8_t *const)>mem_read_cb; |     std::function<iss::status(phys_addr_t, unsigned, uint8_t *const)>mem_read_cb; | ||||||
| @@ -430,37 +303,71 @@ protected: | |||||||
| template <typename BASE> | template <typename BASE> | ||||||
| riscv_hart_m_p<BASE>::riscv_hart_m_p() | riscv_hart_m_p<BASE>::riscv_hart_m_p() | ||||||
| : state() | : state() | ||||||
| , cycle_offset(0) |  | ||||||
| , instr_if(*this) { | , instr_if(*this) { | ||||||
|  |     // reset values | ||||||
|     csr[misa] = traits<BASE>::MISA_VAL; |     csr[misa] = traits<BASE>::MISA_VAL; | ||||||
|  |     csr[mvendorid] = 0x669; | ||||||
|  |     csr[marchid] = 0x80000003; | ||||||
|  |     csr[mimpid] = 1; | ||||||
|  |  | ||||||
|     uart_buf.str(""); |     uart_buf.str(""); | ||||||
|     for (unsigned addr = mcycle; addr <= hpmcounter31; ++addr) csr_wr_cb[addr] = nullptr; |     for (unsigned addr = mhpmcounter3; addr <= mhpmcounter31; ++addr){ | ||||||
|     for (unsigned addr = mcycleh; addr <= hpmcounter31h; ++addr) csr_wr_cb[addr] = nullptr; |         csr_rd_cb[addr] = &this_class::read_null; | ||||||
|     // special handling |         csr_wr_cb[addr] = &this_class::write_reg; | ||||||
|     csr_rd_cb[time] = &riscv_hart_m_p<BASE>::read_time; |  | ||||||
|     csr_wr_cb[time] = nullptr; |  | ||||||
|     csr_rd_cb[timeh] = &riscv_hart_m_p<BASE>::read_time; |  | ||||||
|     csr_wr_cb[timeh] = nullptr; |  | ||||||
|     csr_rd_cb[mcycle] = &riscv_hart_m_p<BASE>::read_cycle; |  | ||||||
|     csr_rd_cb[mcycleh] = &riscv_hart_m_p<BASE>::read_cycle; |  | ||||||
|     csr_rd_cb[minstret] = &riscv_hart_m_p<BASE>::read_cycle; |  | ||||||
|     csr_rd_cb[minstreth] = &riscv_hart_m_p<BASE>::read_cycle; |  | ||||||
|     csr_rd_cb[mstatus] = &riscv_hart_m_p<BASE>::read_status; |  | ||||||
|     csr_wr_cb[mstatus] = &riscv_hart_m_p<BASE>::write_status; |  | ||||||
|     csr_rd_cb[mip] = &riscv_hart_m_p<BASE>::read_ip; |  | ||||||
|     csr_wr_cb[mip] = &riscv_hart_m_p<BASE>::write_ip; |  | ||||||
|     csr_rd_cb[mie] = &riscv_hart_m_p<BASE>::read_ie; |  | ||||||
|     csr_wr_cb[mie] = &riscv_hart_m_p<BASE>::write_ie; |  | ||||||
|     csr_rd_cb[mhartid] = &riscv_hart_m_p<BASE>::read_hartid; |  | ||||||
|     // common regs |  | ||||||
|     const std::array<unsigned, 6> addrs{{mepc, mtvec, mscratch, mcause, mtval, mscratch}}; |  | ||||||
|     for(auto addr: addrs) { |  | ||||||
|         csr_rd_cb[addr] = &riscv_hart_m_p<BASE>::read_reg; |  | ||||||
|         csr_wr_cb[addr] = &riscv_hart_m_p<BASE>::write_reg; |  | ||||||
|     } |     } | ||||||
|     // read-only registers |     for (unsigned addr = mhpmcounter3h; addr <= mhpmcounter31h; ++addr){ | ||||||
|     csr_rd_cb[misa] = &riscv_hart_m_p<BASE>::read_reg; |         csr_rd_cb[addr] = &this_class::read_null; | ||||||
|     csr_wr_cb[misa] = nullptr; |         csr_wr_cb[addr] = &this_class::write_reg; | ||||||
|  |     } | ||||||
|  |     for (unsigned addr = mhpmevent3; addr <= mhpmevent31; ++addr){ | ||||||
|  |         csr_rd_cb[addr] = &this_class::read_null; | ||||||
|  |         csr_wr_cb[addr] = &this_class::write_reg; | ||||||
|  |     } | ||||||
|  |     for (unsigned addr = hpmcounter3; addr <= hpmcounter31; ++addr){ | ||||||
|  |         csr_rd_cb[addr] = &this_class::read_null; | ||||||
|  |     } | ||||||
|  |     for (unsigned addr = hpmcounter3h; addr <= hpmcounter31h; ++addr){ | ||||||
|  |         csr_rd_cb[addr] = &this_class::read_null; | ||||||
|  |         //csr_wr_cb[addr] = &this_class::write_reg; | ||||||
|  |     } | ||||||
|  |     // common regs | ||||||
|  |     const std::array<unsigned, 10> addrs{{misa, mvendorid, marchid, mimpid, mepc, mtvec, mscratch, mcause, mtval, mscratch}}; | ||||||
|  |     for(auto addr: addrs) { | ||||||
|  |         csr_rd_cb[addr] = &this_class::read_reg; | ||||||
|  |         csr_wr_cb[addr] = &this_class::write_reg; | ||||||
|  |     } | ||||||
|  |     // special handling & overrides | ||||||
|  |     csr_rd_cb[time] = &this_class::read_time; | ||||||
|  |     csr_rd_cb[timeh] = &this_class::read_time; | ||||||
|  |     csr_rd_cb[cycle] = &this_class::read_cycle; | ||||||
|  |     csr_rd_cb[cycleh] = &this_class::read_cycle; | ||||||
|  |     csr_rd_cb[instret] = &this_class::read_instret; | ||||||
|  |     csr_rd_cb[instreth] = &this_class::read_instret; | ||||||
|  |  | ||||||
|  |     csr_rd_cb[mcycle] = &this_class::read_cycle; | ||||||
|  |     csr_wr_cb[mcycle] = &this_class::write_cycle; | ||||||
|  |     csr_rd_cb[mcycleh] = &this_class::read_cycle; | ||||||
|  |     csr_wr_cb[mcycleh] = &this_class::write_cycle; | ||||||
|  |     csr_rd_cb[minstret] = &this_class::read_instret; | ||||||
|  |     csr_wr_cb[minstret] = &this_class::write_instret; | ||||||
|  |     csr_rd_cb[minstreth] = &this_class::read_instret; | ||||||
|  |     csr_wr_cb[minstreth] = &this_class::write_instret; | ||||||
|  |     csr_rd_cb[mstatus] = &this_class::read_status; | ||||||
|  |     csr_wr_cb[mstatus] = &this_class::write_status; | ||||||
|  |     csr_wr_cb[mcause] = &this_class::write_cause; | ||||||
|  |     csr_rd_cb[mtvec] = &this_class::read_tvec; | ||||||
|  |     csr_wr_cb[mepc] = &this_class::write_epc; | ||||||
|  |     csr_rd_cb[mip] = &this_class::read_ip; | ||||||
|  |     csr_wr_cb[mip] = &this_class::write_ip; | ||||||
|  |     csr_rd_cb[mie] = &this_class::read_ie; | ||||||
|  |     csr_wr_cb[mie] = &this_class::write_ie; | ||||||
|  |     csr_rd_cb[mhartid] = &this_class::read_hartid; | ||||||
|  |     csr_rd_cb[mcounteren] = &this_class::read_null; | ||||||
|  |     csr_wr_cb[mcounteren] = &this_class::write_null; | ||||||
|  |     csr_wr_cb[misa] = &this_class::write_null; | ||||||
|  |     csr_wr_cb[mvendorid] = &this_class::write_null; | ||||||
|  |     csr_wr_cb[marchid] = &this_class::write_null; | ||||||
|  |     csr_wr_cb[mimpid] = &this_class::write_null; | ||||||
| } | } | ||||||
|  |  | ||||||
| template <typename BASE> std::pair<uint64_t, bool> riscv_hart_m_p<BASE>::load_file(std::string name, int type) { | template <typename BASE> std::pair<uint64_t, bool> riscv_hart_m_p<BASE>::load_file(std::string name, int type) { | ||||||
| @@ -481,6 +388,7 @@ template <typename BASE> std::pair<uint64_t, bool> riscv_hart_m_p<BASE>::load_fi | |||||||
|                 if (sizeof(reg_t) == 4) throw std::runtime_error("wrong elf class in file"); |                 if (sizeof(reg_t) == 4) throw std::runtime_error("wrong elf class in file"); | ||||||
|             if (reader.get_type() != ET_EXEC) throw std::runtime_error("wrong elf type in file"); |             if (reader.get_type() != ET_EXEC) throw std::runtime_error("wrong elf type in file"); | ||||||
|             if (reader.get_machine() != EM_RISCV) throw std::runtime_error("wrong elf machine in file"); |             if (reader.get_machine() != EM_RISCV) throw std::runtime_error("wrong elf machine in file"); | ||||||
|  |             auto entry = reader.get_entry(); | ||||||
|             for (const auto pseg : reader.segments) { |             for (const auto pseg : reader.segments) { | ||||||
|                 const auto fsize = pseg->get_file_size(); // 0x42c/0x0 |                 const auto fsize = pseg->get_file_size(); // 0x42c/0x0 | ||||||
|                 const auto seg_data = pseg->get_data(); |                 const auto seg_data = pseg->get_data(); | ||||||
| @@ -489,18 +397,39 @@ template <typename BASE> std::pair<uint64_t, bool> riscv_hart_m_p<BASE>::load_fi | |||||||
|                             traits<BASE>::MEM, pseg->get_physical_address(), |                             traits<BASE>::MEM, pseg->get_physical_address(), | ||||||
|                             fsize, reinterpret_cast<const uint8_t *const>(seg_data)); |                             fsize, reinterpret_cast<const uint8_t *const>(seg_data)); | ||||||
|                     if (res != iss::Ok) |                     if (res != iss::Ok) | ||||||
|                         LOG(ERROR) << "problem writing " << fsize << "bytes to 0x" << std::hex |                         LOG(ERR) << "problem writing " << fsize << "bytes to 0x" << std::hex | ||||||
|                                    << pseg->get_physical_address(); |                                    << pseg->get_physical_address(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             for(const auto sec : reader.sections) { |             for(const auto sec : reader.sections) { | ||||||
|                 if (sec->get_name() == ".tohost") { |                 if(sec->get_name() == ".symtab") { | ||||||
|  |                     if ( SHT_SYMTAB == sec->get_type() || | ||||||
|  |                             SHT_DYNSYM == sec->get_type() ) { | ||||||
|  |                         ELFIO::symbol_section_accessor symbols( reader, sec ); | ||||||
|  |                         auto sym_no = symbols.get_symbols_num(); | ||||||
|  |                         std::string   name; | ||||||
|  |                         ELFIO::Elf64_Addr    value   = 0; | ||||||
|  |                         ELFIO::Elf_Xword     size    = 0; | ||||||
|  |                         unsigned char bind    = 0; | ||||||
|  |                         unsigned char type    = 0; | ||||||
|  |                         ELFIO::Elf_Half      section = 0; | ||||||
|  |                         unsigned char other   = 0; | ||||||
|  |                         for ( auto i = 0U; i < sym_no; ++i ) { | ||||||
|  |                             symbols.get_symbol( i, name, value, size, bind, type, section, other ); | ||||||
|  |                             if(name=="tohost") { | ||||||
|  |                                 tohost = value; | ||||||
|  |                             } else if(name=="fromhost") { | ||||||
|  |                                 fromhost = value; | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } else if (sec->get_name() == ".tohost") { | ||||||
|                     tohost = sec->get_address(); |                     tohost = sec->get_address(); | ||||||
|                     fromhost = tohost + 0x40; |                     fromhost = tohost + 0x40; | ||||||
|                 } |                 } | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return std::make_pair(reader.get_entry(), true); |             } | ||||||
|  |             return std::make_pair(entry, true); | ||||||
|         } |         } | ||||||
|         throw std::runtime_error("memory load file is not a valid elf file"); |         throw std::runtime_error("memory load file is not a valid elf file"); | ||||||
|     } |     } | ||||||
| @@ -529,13 +458,23 @@ iss::status riscv_hart_m_p<BASE>::read(const address_type type, const access_typ | |||||||
|                 return iss::Err; |                 return iss::Err; | ||||||
|             } |             } | ||||||
|             try { |             try { | ||||||
|  |                 auto alignment = access == iss::access_type::FETCH? (traits<BASE>::MISA_VAL&0x100? 2 : 4) : length; | ||||||
|  |                 if(alignment>1 && (addr&(alignment-1))){ | ||||||
|  |                     this->reg.trap_state = 1<<31 | 4<<16; | ||||||
|  |                     fault_data=addr; | ||||||
|  |                     return iss::Err; | ||||||
|  |                 } | ||||||
|                 auto res = type==iss::address_type::PHYSICAL? |                 auto res = type==iss::address_type::PHYSICAL? | ||||||
|                         read_mem( BASE::v2p(phys_addr_t{access, space, addr}), length, data): |                         read_mem( BASE::v2p(phys_addr_t{access, space, addr}), length, data): | ||||||
|                         read_mem( BASE::v2p(iss::addr_t{access, type, space, addr}), length, data); |                         read_mem( BASE::v2p(iss::addr_t{access, type, space, addr}), length, data); | ||||||
|                 if (unlikely(res != iss::Ok)) this->reg.trap_state = (1 << 31) | (5 << 16); // issue trap 5 (load access fault |                 if (unlikely(res != iss::Ok)){ | ||||||
|  |                     this->reg.trap_state = (1 << 31) | (5 << 16); // issue trap 5 (load access fault | ||||||
|  |                     fault_data=addr; | ||||||
|  |                 } | ||||||
|                 return res; |                 return res; | ||||||
|             } catch (trap_access &ta) { |             } catch (trap_access &ta) { | ||||||
|                 this->reg.trap_state = (1 << 31) | ta.id; |                 this->reg.trap_state = (1 << 31) | ta.id; | ||||||
|  |                 fault_data=ta.addr; | ||||||
|                 return iss::Err; |                 return iss::Err; | ||||||
|             } |             } | ||||||
|         } break; |         } break; | ||||||
| @@ -561,6 +500,7 @@ iss::status riscv_hart_m_p<BASE>::read(const address_type type, const access_typ | |||||||
|         return iss::Ok; |         return iss::Ok; | ||||||
|     } catch (trap_access &ta) { |     } catch (trap_access &ta) { | ||||||
|         this->reg.trap_state = (1 << 31) | ta.id; |         this->reg.trap_state = (1 << 31) | ta.id; | ||||||
|  |         fault_data=ta.addr; | ||||||
|         return iss::Err; |         return iss::Err; | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -601,14 +541,22 @@ iss::status riscv_hart_m_p<BASE>::write(const address_type type, const access_ty | |||||||
|                 return iss::Err; |                 return iss::Err; | ||||||
|             } |             } | ||||||
|             try { |             try { | ||||||
|  |                 if(!(access && iss::access_type::DEBUG) &&  length>1 && (addr&(length-1))){ | ||||||
|  |                     this->reg.trap_state = 1<<31 | 6<<16; | ||||||
|  |                     fault_data=addr; | ||||||
|  |                     return iss::Err; | ||||||
|  |                 } | ||||||
|                 auto res = type==iss::address_type::PHYSICAL? |                 auto res = type==iss::address_type::PHYSICAL? | ||||||
|                         write_mem(phys_addr_t{access, space, addr}, length, data): |                         write_mem(phys_addr_t{access, space, addr}, length, data): | ||||||
|                         write_mem(BASE::v2p(iss::addr_t{access, type, space, addr}), length, data); |                         write_mem(BASE::v2p(iss::addr_t{access, type, space, addr}), length, data); | ||||||
|                 if (unlikely(res != iss::Ok)) |                 if (unlikely(res != iss::Ok)) { | ||||||
|                     this->reg.trap_state = (1 << 31) | (5 << 16); // issue trap 7 (Store/AMO access fault) |                     this->reg.trap_state = (1 << 31) | (7 << 16); // issue trap 7 (Store/AMO access fault) | ||||||
|  |                     fault_data=addr; | ||||||
|  |                 } | ||||||
|                 return res; |                 return res; | ||||||
|             } catch (trap_access &ta) { |             } catch (trap_access &ta) { | ||||||
|                 this->reg.trap_state = (1 << 31) | ta.id; |                 this->reg.trap_state = (1 << 31) | ta.id; | ||||||
|  |                 fault_data=ta.addr; | ||||||
|                 return iss::Err; |                 return iss::Err; | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -668,6 +616,7 @@ iss::status riscv_hart_m_p<BASE>::write(const address_type type, const access_ty | |||||||
|         return iss::Ok; |         return iss::Ok; | ||||||
|     } catch (trap_access &ta) { |     } catch (trap_access &ta) { | ||||||
|         this->reg.trap_state = (1 << 31) | ta.id; |         this->reg.trap_state = (1 << 31) | ta.id; | ||||||
|  |         fault_data=ta.addr; | ||||||
|         return iss::Err; |         return iss::Err; | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -701,6 +650,11 @@ template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_reg(unsigned add | |||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_null(unsigned addr, reg_t &val) { | ||||||
|  |     val = 0; | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
| template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_reg(unsigned addr, reg_t val) { | template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_reg(unsigned addr, reg_t val) { | ||||||
|     csr[addr] = val; |     csr[addr] = val; | ||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| @@ -717,8 +671,50 @@ template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_cycle(unsigned a | |||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_cycle(unsigned addr, reg_t val) { | ||||||
|  |     if (sizeof(typename traits<BASE>::reg_t) != 4) { | ||||||
|  |         if (addr == mcycleh) | ||||||
|  |             return iss::Err; | ||||||
|  |         mcycle_csr = static_cast<uint64_t>(val); | ||||||
|  |     } else { | ||||||
|  |         if (addr == mcycle) { | ||||||
|  |             mcycle_csr = (mcycle_csr & 0xffffffff00000000) + val; | ||||||
|  |         } else  { | ||||||
|  |             mcycle_csr = (static_cast<uint64_t>(val)<<32) + (mcycle_csr & 0xffffffff); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     cycle_offset = mcycle_csr-this->reg.icount; // TODO: relying on wrap-around | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_instret(unsigned addr, reg_t &val) { | ||||||
|  |     if ((addr&0xff) == (minstret&0xff)) { | ||||||
|  |         val = static_cast<reg_t>(this->reg.instret); | ||||||
|  |     } else if ((addr&0xff) == (minstreth&0xff)) { | ||||||
|  |         if (sizeof(typename traits<BASE>::reg_t) != 4) return iss::Err; | ||||||
|  |         val = static_cast<reg_t>(this->reg.instret >> 32); | ||||||
|  |     } | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_instret(unsigned addr, reg_t val) { | ||||||
|  |     if (sizeof(typename traits<BASE>::reg_t) != 4) { | ||||||
|  |         if ((addr&0xff) == (minstreth&0xff)) | ||||||
|  |             return iss::Err; | ||||||
|  |         this->reg.instret = static_cast<uint64_t>(val); | ||||||
|  |     } else { | ||||||
|  |         if ((addr&0xff) == (minstret&0xff)) { | ||||||
|  |             this->reg.instret = (this->reg.instret & 0xffffffff00000000) + val; | ||||||
|  |         } else  { | ||||||
|  |             this->reg.instret = (static_cast<uint64_t>(val)<<32) + (this->reg.instret & 0xffffffff); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     this->reg.instret--; | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
| template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_time(unsigned addr, reg_t &val) { | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_time(unsigned addr, reg_t &val) { | ||||||
|     uint64_t time_val = (this->reg.icount + cycle_offset) / (100000000 / 32768 - 1); //-> ~3052; |     uint64_t time_val = this->reg.icount / (100000000 / 32768 - 1); //-> ~3052; | ||||||
|     if (addr == time) { |     if (addr == time) { | ||||||
|         val = static_cast<reg_t>(time_val); |         val = static_cast<reg_t>(time_val); | ||||||
|     } else if (addr == timeh) { |     } else if (addr == timeh) { | ||||||
| @@ -728,8 +724,13 @@ template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_time(unsigned ad | |||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_tvec(unsigned addr, reg_t &val) { | ||||||
|  |     val = csr[mtvec] & ~2; | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
| template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_status(unsigned addr, reg_t &val) { | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_status(unsigned addr, reg_t &val) { | ||||||
|     val = state.mstatus & hart_state<reg_t>::get_mask(); |     val = state.mstatus & hart_state_type::get_mask(); | ||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -739,9 +740,13 @@ template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_status(unsigned | |||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_cause(unsigned addr, reg_t val) { | ||||||
|  |     csr[mcause] = val & ((1UL<<(traits<BASE>::XLEN-1))|0xf); //TODO: make exception code size configurable | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
| template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_ie(unsigned addr, reg_t &val) { | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_ie(unsigned addr, reg_t &val) { | ||||||
|     val = csr[mie]; |     val = csr[mie]; | ||||||
|     val &= csr[mideleg]; |  | ||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -759,7 +764,6 @@ template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_ie(unsigned add | |||||||
|  |  | ||||||
| template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_ip(unsigned addr, reg_t &val) { | template <typename BASE> iss::status riscv_hart_m_p<BASE>::read_ip(unsigned addr, reg_t &val) { | ||||||
|     val = csr[mip]; |     val = csr[mip]; | ||||||
|     val &= csr[mideleg]; |  | ||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -771,9 +775,13 @@ template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_ip(unsigned add | |||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | template <typename BASE> iss::status riscv_hart_m_p<BASE>::write_epc(unsigned addr, reg_t val) { | ||||||
|  |     csr[addr] = val & get_pc_mask(); | ||||||
|  |     return iss::Ok; | ||||||
|  | } | ||||||
|  |  | ||||||
| template <typename BASE> | template <typename BASE> | ||||||
| iss::status riscv_hart_m_p<BASE>::read_mem(phys_addr_t paddr, unsigned length, uint8_t *const data) { | iss::status riscv_hart_m_p<BASE>::read_mem(phys_addr_t paddr, unsigned length, uint8_t *const data) { | ||||||
|     if ((paddr.val + length) > mem.size()) return iss::Err; |  | ||||||
|     if(mem_read_cb) return mem_read_cb(paddr, length, data); |     if(mem_read_cb) return mem_read_cb(paddr, length, data); | ||||||
|     switch (paddr.val) { |     switch (paddr.val) { | ||||||
|     case 0x0200BFF8: { // CLINT base, mtime reg |     case 0x0200BFF8: { // CLINT base, mtime reg | ||||||
| @@ -789,9 +797,9 @@ iss::status riscv_hart_m_p<BASE>::read_mem(phys_addr_t paddr, unsigned length, u | |||||||
|         if (this->reg.icount > 30000) data[3] |= 0x80; |         if (this->reg.icount > 30000) data[3] |= 0x80; | ||||||
|     } break; |     } break; | ||||||
|     default: { |     default: { | ||||||
|         const auto &p = mem(paddr.val / mem.page_size); |         for(auto offs=0U; offs<length; ++offs) { | ||||||
|         auto offs = paddr.val & mem.page_addr_mask; |             *(data + offs)=mem[(paddr.val+offs)%mem.size()]; | ||||||
|         std::copy(p.data() + offs, p.data() + offs + length, data); |         } | ||||||
|     } |     } | ||||||
|     } |     } | ||||||
|     return iss::Ok; |     return iss::Ok; | ||||||
| @@ -799,7 +807,6 @@ iss::status riscv_hart_m_p<BASE>::read_mem(phys_addr_t paddr, unsigned length, u | |||||||
|  |  | ||||||
| template <typename BASE> | template <typename BASE> | ||||||
| iss::status riscv_hart_m_p<BASE>::write_mem(phys_addr_t paddr, unsigned length, const uint8_t *const data) { | iss::status riscv_hart_m_p<BASE>::write_mem(phys_addr_t paddr, unsigned length, const uint8_t *const data) { | ||||||
|     if ((paddr.val + length) > mem.size()) return iss::Err; |  | ||||||
|     if(mem_write_cb) return mem_write_cb(paddr, length, data); |     if(mem_write_cb) return mem_write_cb(paddr, length, data); | ||||||
|     switch (paddr.val) { |     switch (paddr.val) { | ||||||
|     case 0x10013000: // UART0 base, TXFIFO reg |     case 0x10013000: // UART0 base, TXFIFO reg | ||||||
| @@ -831,9 +838,10 @@ iss::status riscv_hart_m_p<BASE>::write_mem(phys_addr_t paddr, unsigned length, | |||||||
|         std::copy(data, data + length, p.data() + (paddr.val & mem.page_addr_mask)); |         std::copy(data, data + length, p.data() + (paddr.val & mem.page_addr_mask)); | ||||||
|         // tohost handling in case of riscv-test |         // tohost handling in case of riscv-test | ||||||
|         if (paddr.access && iss::access_type::FUNC) { |         if (paddr.access && iss::access_type::FUNC) { | ||||||
|             auto tohost_upper = (traits<BASE>::XLEN == 32 && paddr.val == (tohost + 4)); |             auto tohost_upper = (traits<BASE>::XLEN == 32 && paddr.val == (tohost + 4)) || | ||||||
|  |                                 (traits<BASE>::XLEN == 64 && paddr.val == tohost); | ||||||
|             auto tohost_lower = |             auto tohost_lower = | ||||||
|                 (traits<BASE>::XLEN == 32 && paddr.val == tohost); |                 (traits<BASE>::XLEN == 32 && paddr.val == tohost) || (traits<BASE>::XLEN == 64 && paddr.val == tohost); | ||||||
|             if (tohost_lower || tohost_upper) { |             if (tohost_lower || tohost_upper) { | ||||||
|                 uint64_t hostvar = *reinterpret_cast<uint64_t *>(p.data() + (tohost & mem.page_addr_mask)); |                 uint64_t hostvar = *reinterpret_cast<uint64_t *>(p.data() + (tohost & mem.page_addr_mask)); | ||||||
|                 if (tohost_upper || (tohost_lower && to_host_wr_cnt > 0)) { |                 if (tohost_upper || (tohost_lower && to_host_wr_cnt > 0)) { | ||||||
| @@ -864,7 +872,8 @@ iss::status riscv_hart_m_p<BASE>::write_mem(phys_addr_t paddr, unsigned length, | |||||||
|                     } |                     } | ||||||
|                 } else if (tohost_lower) |                 } else if (tohost_lower) | ||||||
|                     to_host_wr_cnt++; |                     to_host_wr_cnt++; | ||||||
|             } else if (traits<BASE>::XLEN == 32 && paddr.val == fromhost + 4) { |             } else if ((traits<BASE>::XLEN == 32 && paddr.val == fromhost + 4) || | ||||||
|  |                        (traits<BASE>::XLEN == 64 && paddr.val == fromhost)) { | ||||||
|                 uint64_t fhostvar = *reinterpret_cast<uint64_t *>(p.data() + (fromhost & mem.page_addr_mask)); |                 uint64_t fhostvar = *reinterpret_cast<uint64_t *>(p.data() + (fromhost & mem.page_addr_mask)); | ||||||
|                 *reinterpret_cast<uint64_t *>(p.data() + (tohost & mem.page_addr_mask)) = fhostvar; |                 *reinterpret_cast<uint64_t *>(p.data() + (tohost & mem.page_addr_mask)) = fhostvar; | ||||||
|             } |             } | ||||||
| @@ -876,11 +885,11 @@ iss::status riscv_hart_m_p<BASE>::write_mem(phys_addr_t paddr, unsigned length, | |||||||
|  |  | ||||||
| template <typename BASE> inline void riscv_hart_m_p<BASE>::reset(uint64_t address) { | template <typename BASE> inline void riscv_hart_m_p<BASE>::reset(uint64_t address) { | ||||||
|     BASE::reset(address); |     BASE::reset(address); | ||||||
|     state.mstatus = hart_state<reg_t>::mstatus_reset_val; |     state.mstatus = hart_state_type::mstatus_reset_val; | ||||||
| } | } | ||||||
|  |  | ||||||
| template <typename BASE> void riscv_hart_m_p<BASE>::check_interrupt() { | template <typename BASE> void riscv_hart_m_p<BASE>::check_interrupt() { | ||||||
|     auto ideleg = csr[mideleg]; |     //auto ideleg = csr[mideleg]; | ||||||
|     // Multiple simultaneous interrupts and traps at the same privilege level are |     // Multiple simultaneous interrupts and traps at the same privilege level are | ||||||
|     // handled in the following decreasing priority order: |     // handled in the following decreasing priority order: | ||||||
|     // external interrupts, software interrupts, timer interrupts, then finally |     // external interrupts, software interrupts, timer interrupts, then finally | ||||||
| @@ -889,7 +898,7 @@ template <typename BASE> void riscv_hart_m_p<BASE>::check_interrupt() { | |||||||
|  |  | ||||||
|     bool mie = state.mstatus.MIE; |     bool mie = state.mstatus.MIE; | ||||||
|     auto m_enabled = this->reg.PRIV < PRIV_M || (this->reg.PRIV == PRIV_M && mie); |     auto m_enabled = this->reg.PRIV < PRIV_M || (this->reg.PRIV == PRIV_M && mie); | ||||||
|     auto enabled_interrupts = m_enabled ? ena_irq & ~ideleg : 0; |     auto enabled_interrupts = m_enabled ? ena_irq : 0; | ||||||
|  |  | ||||||
|     if (enabled_interrupts != 0) { |     if (enabled_interrupts != 0) { | ||||||
|         int res = 0; |         int res = 0; | ||||||
| @@ -901,7 +910,7 @@ template <typename BASE> void riscv_hart_m_p<BASE>::check_interrupt() { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| template <typename BASE> uint64_t riscv_hart_m_p<BASE>::enter_trap(uint64_t flags, uint64_t addr) { | template <typename BASE> uint64_t riscv_hart_m_p<BASE>::enter_trap(uint64_t flags, uint64_t addr, uint64_t instr) { | ||||||
|     // flags are ACTIVE[31:31], CAUSE[30:16], TRAPID[15:0] |     // flags are ACTIVE[31:31], CAUSE[30:16], TRAPID[15:0] | ||||||
|     // calculate and write mcause val |     // calculate and write mcause val | ||||||
|     auto trap_id = bit_sub<0, 16>(flags); |     auto trap_id = bit_sub<0, 16>(flags); | ||||||
| @@ -910,11 +919,11 @@ template <typename BASE> uint64_t riscv_hart_m_p<BASE>::enter_trap(uint64_t flag | |||||||
|     // calculate effective privilege level |     // calculate effective privilege level | ||||||
|     if (trap_id == 0) { // exception |     if (trap_id == 0) { // exception | ||||||
|         // store ret addr in xepc register |         // store ret addr in xepc register | ||||||
|         csr[mepc] = static_cast<reg_t>(addr); // store actual address instruction of exception |         csr[mepc] = static_cast<reg_t>(addr) & get_pc_mask(); // store actual address instruction of exception | ||||||
|         csr[mtval] = fault_data; |         csr[mtval] = cause==2?((instr & 0x3)==3?instr:instr&0xffff):fault_data; | ||||||
|         fault_data = 0; |         fault_data = 0; | ||||||
|     } else { |     } else { | ||||||
|         csr[mepc] = this->reg.NEXT_PC; // store next address if interrupt |         csr[mepc] = this->reg.NEXT_PC & get_pc_mask(); // store next address if interrupt | ||||||
|         this->reg.pending_trap = 0; |         this->reg.pending_trap = 0; | ||||||
|     } |     } | ||||||
|     csr[mcause] = (trap_id << 31) + cause; |     csr[mcause] = (trap_id << 31) + cause; | ||||||
| @@ -933,13 +942,17 @@ template <typename BASE> uint64_t riscv_hart_m_p<BASE>::enter_trap(uint64_t flag | |||||||
|     auto ivec = csr[mtvec]; |     auto ivec = csr[mtvec]; | ||||||
|     // calculate addr// set NEXT_PC to trap addressess to jump to based on MODE |     // calculate addr// set NEXT_PC to trap addressess to jump to based on MODE | ||||||
|     // bits in mtvec |     // bits in mtvec | ||||||
|     this->reg.NEXT_PC = ivec & ~0x1UL; |     this->reg.NEXT_PC = ivec & ~0x3UL; | ||||||
|     if ((ivec & 0x1) == 1 && trap_id != 0) this->reg.NEXT_PC += 4 * cause; |     if ((ivec & 0x1) == 1 && trap_id != 0) this->reg.NEXT_PC += 4 * cause; | ||||||
|     // reset trap state |     // reset trap state | ||||||
|     this->reg.PRIV = PRIV_M; |     this->reg.PRIV = PRIV_M; | ||||||
|     this->reg.trap_state = 0; |     this->reg.trap_state = 0; | ||||||
|     std::array<char, 32> buffer; |     std::array<char, 32> buffer; | ||||||
|  | #if defined(_MSC_VER) | ||||||
|  |     sprintf(buffer.data(), "0x%016llx", addr); | ||||||
|  | #else | ||||||
|     sprintf(buffer.data(), "0x%016lx", addr); |     sprintf(buffer.data(), "0x%016lx", addr); | ||||||
|  | #endif | ||||||
|     if((flags&0xffffffff) != 0xffffffff) |     if((flags&0xffffffff) != 0xffffffff) | ||||||
|     CLOG(INFO, disass) << (trap_id ? "Interrupt" : "Trap") << " with cause '" |     CLOG(INFO, disass) << (trap_id ? "Interrupt" : "Trap") << " with cause '" | ||||||
|                        << (trap_id ? irq_str[cause] : trap_str[cause]) << "' (" << cause << ")" |                        << (trap_id ? irq_str[cause] : trap_str[cause]) << "' (" << cause << ")" | ||||||
| @@ -949,13 +962,15 @@ template <typename BASE> uint64_t riscv_hart_m_p<BASE>::enter_trap(uint64_t flag | |||||||
|  |  | ||||||
| template <typename BASE> uint64_t riscv_hart_m_p<BASE>::leave_trap(uint64_t flags) { | template <typename BASE> uint64_t riscv_hart_m_p<BASE>::leave_trap(uint64_t flags) { | ||||||
|     state.mstatus.MIE = state.mstatus.MPIE; |     state.mstatus.MIE = state.mstatus.MPIE; | ||||||
|  |     state.mstatus.MPIE = 1; | ||||||
|     // sets the pc to the value stored in the x epc register. |     // sets the pc to the value stored in the x epc register. | ||||||
|     this->reg.NEXT_PC = csr[mepc]; |     this->reg.NEXT_PC = csr[mepc] & get_pc_mask(); | ||||||
|     CLOG(INFO, disass) << "Executing xRET"; |     CLOG(INFO, disass) << "Executing xRET"; | ||||||
|  |     check_interrupt(); | ||||||
|     return this->reg.NEXT_PC; |     return this->reg.NEXT_PC; | ||||||
| } | } | ||||||
|  |  | ||||||
| } // namespace arch | } // namespace arch | ||||||
| } // namespace iss | } // namespace iss | ||||||
|  |  | ||||||
| #endif /* _RISCV_CORE_H_ */ | #endif /* _RISCV_HART_M_P_H */ | ||||||
|   | |||||||
							
								
								
									
										1388
									
								
								incl/iss/arch/riscv_hart_msu_vp.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1388
									
								
								incl/iss/arch/riscv_hart_msu_vp.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										1286
									
								
								incl/iss/arch/riscv_hart_mu_p.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1286
									
								
								incl/iss/arch/riscv_hart_mu_p.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -30,8 +30,8 @@ | |||||||
|  * |  * | ||||||
|  *******************************************************************************/ |  *******************************************************************************/ | ||||||
| 
 | 
 | ||||||
| #ifndef _TGF_C_H_ | #ifndef _TGC_C_H_ | ||||||
| #define _TGF_C_H_ | #define _TGC_C_H_ | ||||||
| 
 | 
 | ||||||
| #include <array> | #include <array> | ||||||
| #include <iss/arch/traits.h> | #include <iss/arch/traits.h> | ||||||
| @@ -41,11 +41,11 @@ | |||||||
| namespace iss { | namespace iss { | ||||||
| namespace arch { | namespace arch { | ||||||
| 
 | 
 | ||||||
| struct tgf_c; | struct tgc_c; | ||||||
| 
 | 
 | ||||||
| template <> struct traits<tgf_c> { | template <> struct traits<tgc_c> { | ||||||
| 
 | 
 | ||||||
|     constexpr static char const* const core_type = "TGF_C"; |     constexpr static char const* const core_type = "TGC_C"; | ||||||
|      |      | ||||||
|     static constexpr std::array<const char*, 35> reg_names{ |     static constexpr std::array<const char*, 35> reg_names{ | ||||||
|         {"X0", "X1", "X2", "X3", "X4", "X5", "X6", "X7", "X8", "X9", "X10", "X11", "X12", "X13", "X14", "X15", "X16", "X17", "X18", "X19", "X20", "X21", "X22", "X23", "X24", "X25", "X26", "X27", "X28", "X29", "X30", "X31", "PC", "NEXT_PC", "PRIV"}}; |         {"X0", "X1", "X2", "X3", "X4", "X5", "X6", "X7", "X8", "X9", "X10", "X11", "X12", "X13", "X14", "X15", "X16", "X17", "X18", "X19", "X20", "X21", "X22", "X23", "X24", "X25", "X26", "X27", "X28", "X29", "X30", "X31", "PC", "NEXT_PC", "PRIV"}}; | ||||||
| @@ -53,7 +53,7 @@ template <> struct traits<tgf_c> { | |||||||
|     static constexpr std::array<const char*, 35> reg_aliases{ |     static constexpr std::array<const char*, 35> reg_aliases{ | ||||||
|         {"X0", "X1", "X2", "X3", "X4", "X5", "X6", "X7", "X8", "X9", "X10", "X11", "X12", "X13", "X14", "X15", "X16", "X17", "X18", "X19", "X20", "X21", "X22", "X23", "X24", "X25", "X26", "X27", "X28", "X29", "X30", "X31", "PC", "NEXT_PC", "PRIV"}}; |         {"X0", "X1", "X2", "X3", "X4", "X5", "X6", "X7", "X8", "X9", "X10", "X11", "X12", "X13", "X14", "X15", "X16", "X17", "X18", "X19", "X20", "X21", "X22", "X23", "X24", "X25", "X26", "X27", "X28", "X29", "X30", "X31", "PC", "NEXT_PC", "PRIV"}}; | ||||||
| 
 | 
 | ||||||
|     enum constants {XLEN=32, PCLEN=32, MISA_VAL=0b01000000000000000001000100000100, CSR_SIZE=4096, fence=0, fencei=1, fencevmal=2, fencevmau=3, eei_aligned_addresses=1, MUL_LEN=64}; |     enum constants {XLEN=32, PCLEN=32, MISA_VAL=0b01000000000000000001000100000100, PGSIZE=0x1000, PGMASK=0b111111111111, CSR_SIZE=4096, fence=0, fencei=1, fencevmal=2, fencevmau=3, MUL_LEN=64}; | ||||||
| 
 | 
 | ||||||
|     constexpr static unsigned FP_REGS_SIZE = 0; |     constexpr static unsigned FP_REGS_SIZE = 0; | ||||||
| 
 | 
 | ||||||
| @@ -61,7 +61,9 @@ template <> struct traits<tgf_c> { | |||||||
|         X0, X1, X2, X3, X4, X5, X6, X7, X8, X9, X10, X11, X12, X13, X14, X15, X16, X17, X18, X19, X20, X21, X22, X23, X24, X25, X26, X27, X28, X29, X30, X31, PC, NEXT_PC, PRIV, NUM_REGS, |         X0, X1, X2, X3, X4, X5, X6, X7, X8, X9, X10, X11, X12, X13, X14, X15, X16, X17, X18, X19, X20, X21, X22, X23, X24, X25, X26, X27, X28, X29, X30, X31, PC, NEXT_PC, PRIV, NUM_REGS, | ||||||
|         TRAP_STATE=NUM_REGS, |         TRAP_STATE=NUM_REGS, | ||||||
|         PENDING_TRAP, |         PENDING_TRAP, | ||||||
|         ICOUNT |         ICOUNT, | ||||||
|  |         CYCLE, | ||||||
|  |         INSTRET | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     using reg_t = uint32_t; |     using reg_t = uint32_t; | ||||||
| @@ -74,11 +76,11 @@ template <> struct traits<tgf_c> { | |||||||
| 
 | 
 | ||||||
|     using phys_addr_t = iss::typed_addr_t<iss::address_type::PHYSICAL>; |     using phys_addr_t = iss::typed_addr_t<iss::address_type::PHYSICAL>; | ||||||
| 
 | 
 | ||||||
|     static constexpr std::array<const uint32_t, 38> reg_bit_widths{ |     static constexpr std::array<const uint32_t, 40> reg_bit_widths{ | ||||||
|         {32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,8,32,32,64}}; |         {32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,32,8,32,32,64,64,64}}; | ||||||
| 
 | 
 | ||||||
|     static constexpr std::array<const uint32_t, 38> reg_byte_offsets{ |     static constexpr std::array<const uint32_t, 40> reg_byte_offsets{ | ||||||
|         {0,4,8,12,16,20,24,28,32,36,40,44,48,52,56,60,64,68,72,76,80,84,88,92,96,100,104,108,112,116,120,124,128,132,136,137,141,145}}; |         {0,4,8,12,16,20,24,28,32,36,40,44,48,52,56,60,64,68,72,76,80,84,88,92,96,100,104,108,112,116,120,124,128,132,136,137,141,145,153,161}}; | ||||||
| 
 | 
 | ||||||
|     static const uint64_t addr_mask = (reg_t(1) << (XLEN - 1)) | ((reg_t(1) << (XLEN - 1)) - 1); |     static const uint64_t addr_mask = (reg_t(1) << (XLEN - 1)) | ((reg_t(1) << (XLEN - 1)) - 1); | ||||||
| 
 | 
 | ||||||
| @@ -125,51 +127,51 @@ template <> struct traits<tgf_c> { | |||||||
|         OR = 35, |         OR = 35, | ||||||
|         AND = 36, |         AND = 36, | ||||||
|         FENCE = 37, |         FENCE = 37, | ||||||
|         FENCE_I = 38, |         ECALL = 38, | ||||||
|         ECALL = 39, |         EBREAK = 39, | ||||||
|         EBREAK = 40, |         URET = 40, | ||||||
|         URET = 41, |         SRET = 41, | ||||||
|         SRET = 42, |         MRET = 42, | ||||||
|         MRET = 43, |         WFI = 43, | ||||||
|         WFI = 44, |         CSRRW = 44, | ||||||
|         SFENCE_VMA = 45, |         CSRRS = 45, | ||||||
|         CSRRW = 46, |         CSRRC = 46, | ||||||
|         CSRRS = 47, |         CSRRWI = 47, | ||||||
|         CSRRC = 48, |         CSRRSI = 48, | ||||||
|         CSRRWI = 49, |         CSRRCI = 49, | ||||||
|         CSRRSI = 50, |         MUL = 50, | ||||||
|         CSRRCI = 51, |         MULH = 51, | ||||||
|         MUL = 52, |         MULHSU = 52, | ||||||
|         MULH = 53, |         MULHU = 53, | ||||||
|         MULHSU = 54, |         DIV = 54, | ||||||
|         MULHU = 55, |         DIVU = 55, | ||||||
|         DIV = 56, |         REM = 56, | ||||||
|         DIVU = 57, |         REMU = 57, | ||||||
|         REM = 58, |         CADDI4SPN = 58, | ||||||
|         REMU = 59, |         CLW = 59, | ||||||
|         CADDI4SPN = 60, |         CSW = 60, | ||||||
|         CLW = 61, |         CADDI = 61, | ||||||
|         CSW = 62, |         CNOP = 62, | ||||||
|         CADDI = 63, |         CJAL = 63, | ||||||
|         CNOP = 64, |         CLI = 64, | ||||||
|         CJAL = 65, |         CLUI = 65, | ||||||
|         CLI = 66, |         CADDI16SP = 66, | ||||||
|         CLUI = 67, |         __reserved_clui = 67, | ||||||
|         CADDI16SP = 68, |         CSRLI = 68, | ||||||
|         CSRLI = 69, |         CSRAI = 69, | ||||||
|         CSRAI = 70, |         CANDI = 70, | ||||||
|         CANDI = 71, |         CSUB = 71, | ||||||
|         CSUB = 72, |         CXOR = 72, | ||||||
|         CXOR = 73, |         COR = 73, | ||||||
|         COR = 74, |         CAND = 74, | ||||||
|         CAND = 75, |         CJ = 75, | ||||||
|         CJ = 76, |         CBEQZ = 76, | ||||||
|         CBEQZ = 77, |         CBNEZ = 77, | ||||||
|         CBNEZ = 78, |         CSLLI = 78, | ||||||
|         CSLLI = 79, |         CLWSP = 79, | ||||||
|         CLWSP = 80, |         CMV = 80, | ||||||
|         CMV = 81, |         CJR = 81, | ||||||
|         CJR = 82, |         __reserved_cmv = 82, | ||||||
|         CADD = 83, |         CADD = 83, | ||||||
|         CJALR = 84, |         CJALR = 84, | ||||||
|         CEBREAK = 85, |         CEBREAK = 85, | ||||||
| @@ -179,15 +181,15 @@ template <> struct traits<tgf_c> { | |||||||
|     }; |     }; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct tgf_c: public arch_if { | struct tgc_c: public arch_if { | ||||||
| 
 | 
 | ||||||
|     using virt_addr_t = typename traits<tgf_c>::virt_addr_t; |     using virt_addr_t = typename traits<tgc_c>::virt_addr_t; | ||||||
|     using phys_addr_t = typename traits<tgf_c>::phys_addr_t; |     using phys_addr_t = typename traits<tgc_c>::phys_addr_t; | ||||||
|     using reg_t =  typename traits<tgf_c>::reg_t; |     using reg_t =  typename traits<tgc_c>::reg_t; | ||||||
|     using addr_t = typename traits<tgf_c>::addr_t; |     using addr_t = typename traits<tgc_c>::addr_t; | ||||||
| 
 | 
 | ||||||
|     tgf_c(); |     tgc_c(); | ||||||
|     ~tgf_c(); |     ~tgc_c(); | ||||||
| 
 | 
 | ||||||
|     void reset(uint64_t address=0) override; |     void reset(uint64_t address=0) override; | ||||||
| 
 | 
 | ||||||
| @@ -208,9 +210,9 @@ struct tgf_c: public arch_if { | |||||||
|     inline uint64_t stop_code() { return interrupt_sim; } |     inline uint64_t stop_code() { return interrupt_sim; } | ||||||
| 
 | 
 | ||||||
|     inline phys_addr_t v2p(const iss::addr_t& addr){ |     inline phys_addr_t v2p(const iss::addr_t& addr){ | ||||||
|         if (addr.space != traits<tgf_c>::MEM || addr.type == iss::address_type::PHYSICAL || |         if (addr.space != traits<tgc_c>::MEM || addr.type == iss::address_type::PHYSICAL || | ||||||
|                 addr_mode[static_cast<uint16_t>(addr.access)&0x3]==address_type::PHYSICAL) { |                 addr_mode[static_cast<uint16_t>(addr.access)&0x3]==address_type::PHYSICAL) { | ||||||
|             return phys_addr_t(addr.access, addr.space, addr.val&traits<tgf_c>::addr_mask); |             return phys_addr_t(addr.access, addr.space, addr.val&traits<tgc_c>::addr_mask); | ||||||
|         } else |         } else | ||||||
|             return virt2phys(addr); |             return virt2phys(addr); | ||||||
|     } |     } | ||||||
| @@ -223,7 +225,7 @@ struct tgf_c: public arch_if { | |||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
| #pragma pack(push, 1) | #pragma pack(push, 1) | ||||||
|     struct TGF_C_regs {  |     struct TGC_C_regs {  | ||||||
|         uint32_t X0 = 0;  |         uint32_t X0 = 0;  | ||||||
|         uint32_t X1 = 0;  |         uint32_t X1 = 0;  | ||||||
|         uint32_t X2 = 0;  |         uint32_t X2 = 0;  | ||||||
| @@ -261,6 +263,8 @@ protected: | |||||||
|         uint8_t PRIV = 0; |         uint8_t PRIV = 0; | ||||||
|         uint32_t trap_state = 0, pending_trap = 0; |         uint32_t trap_state = 0, pending_trap = 0; | ||||||
|         uint64_t icount = 0; |         uint64_t icount = 0; | ||||||
|  |         uint64_t cycle = 0; | ||||||
|  |         uint64_t instret = 0; | ||||||
|         uint32_t last_branch; |         uint32_t last_branch; | ||||||
|     } reg; |     } reg; | ||||||
| #pragma pack(pop) | #pragma pack(pop) | ||||||
| @@ -275,4 +279,4 @@ protected: | |||||||
| 
 | 
 | ||||||
| } | } | ||||||
| }             | }             | ||||||
| #endif /* _TGF_C_H_ */ | #endif /* _TGC_C_H_ */ | ||||||
							
								
								
									
										62
									
								
								incl/iss/factory.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								incl/iss/factory.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | /******************************************************************************* | ||||||
|  |  * Copyright (C) 2021 MINRES Technologies GmbH | ||||||
|  |  * All rights reserved. | ||||||
|  |  * | ||||||
|  |  * Redistribution and use in source and binary forms, with or without | ||||||
|  |  * modification, are permitted provided that the following conditions are met: | ||||||
|  |  * | ||||||
|  |  * 1. Redistributions of source code must retain the above copyright notice, | ||||||
|  |  *    this list of conditions and the following disclaimer. | ||||||
|  |  * | ||||||
|  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | ||||||
|  |  *    this list of conditions and the following disclaimer in the documentation | ||||||
|  |  *    and/or other materials provided with the distribution. | ||||||
|  |  * | ||||||
|  |  * 3. Neither the name of the copyright holder nor the names of its contributors | ||||||
|  |  *    may be used to endorse or promote products derived from this software | ||||||
|  |  *    without specific prior written permission. | ||||||
|  |  * | ||||||
|  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | ||||||
|  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | ||||||
|  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | ||||||
|  |  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE | ||||||
|  |  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | ||||||
|  |  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | ||||||
|  |  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | ||||||
|  |  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | ||||||
|  |  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | ||||||
|  |  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | ||||||
|  |  * POSSIBILITY OF SUCH DAMAGE. | ||||||
|  |  * | ||||||
|  |  *******************************************************************************/ | ||||||
|  |  | ||||||
|  | #ifndef _ISS_FACTORY_H_ | ||||||
|  | #define _ISS_FACTORY_H_ | ||||||
|  |  | ||||||
|  | #include <iss/iss.h> | ||||||
|  |  | ||||||
|  | namespace iss { | ||||||
|  |  | ||||||
|  | using cpu_ptr = std::unique_ptr<iss::arch_if>; | ||||||
|  | using vm_ptr= std::unique_ptr<iss::vm_if>; | ||||||
|  |  | ||||||
|  | template<typename PLAT> | ||||||
|  | std::tuple<cpu_ptr, vm_ptr> create_cpu(std::string const& backend, unsigned gdb_port){ | ||||||
|  |     using core_type = typename PLAT::core; | ||||||
|  |     core_type* lcpu = new PLAT(); | ||||||
|  |     if(backend == "interp") | ||||||
|  |         return {cpu_ptr{lcpu}, vm_ptr{iss::interp::create(lcpu, gdb_port)}}; | ||||||
|  | #ifdef WITH_LLVM | ||||||
|  |     if(backend == "llvm") | ||||||
|  |         return {cpu_ptr{lcpu}, vm_ptr{iss::llvm::create(lcpu, gdb_port)}}; | ||||||
|  | #endif | ||||||
|  | #ifdef WITH_LLVM | ||||||
|  |     if(backend == "tcc") | ||||||
|  |         return {cpu_ptr{lcpu}, vm_ptr{iss::tcc::create(lcpu, gdb_port)}}; | ||||||
|  | #endif | ||||||
|  |     return {nullptr, nullptr}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #endif /* _ISS_FACTORY_H_ */ | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| /******************************************************************************* | /******************************************************************************* | ||||||
|  * Copyright (C) 2017, 2018 MINRES Technologies GmbH |  * Copyright (C) 2017-2021 MINRES Technologies GmbH | ||||||
|  * All rights reserved. |  * All rights reserved. | ||||||
|  * |  * | ||||||
|  * Redistribution and use in source and binary forms, with or without |  * Redistribution and use in source and binary forms, with or without | ||||||
| @@ -30,33 +30,23 @@ | |||||||
|  * |  * | ||||||
|  *******************************************************************************/ |  *******************************************************************************/ | ||||||
|  |  | ||||||
| #ifndef _SYSC_SIFIVE_FE310_H_ | #ifndef _SYSC_CORE_COMPLEX_H_ | ||||||
| #define _SYSC_SIFIVE_FE310_H_ | #define _SYSC_CORE_COMPLEX_H_ | ||||||
|  |  | ||||||
| #include "tlm/scc/initiator_mixin.h" | #include <tlm/scc/initiator_mixin.h> | ||||||
| #include "scc/traceable.h" | #include <scc/traceable.h> | ||||||
| #include "scc/utilities.h" | #include <scc/tick2time.h> | ||||||
| #include "tlm/scc/scv/tlm_rec_initiator_socket.h" | #include <scc/utilities.h> | ||||||
|  | #include <tlm/scc/scv/tlm_rec_initiator_socket.h> | ||||||
|  | #ifdef CWR_SYSTEMC | ||||||
|  | #include <scmlinc/scml_property.h> | ||||||
|  | #else | ||||||
| #include <cci_configuration> | #include <cci_configuration> | ||||||
|  | #endif | ||||||
| #include <tlm> | #include <tlm> | ||||||
| #include <tlm_core/tlm_1/tlm_req_rsp/tlm_1_interfaces/tlm_core_ifs.h> |  | ||||||
| #include <tlm_utils/tlm_quantumkeeper.h> | #include <tlm_utils/tlm_quantumkeeper.h> | ||||||
| #include <util/range_lut.h> | #include <util/range_lut.h> | ||||||
|  | #include <memory> | ||||||
| class scv_tr_db; |  | ||||||
| class scv_tr_stream; |  | ||||||
| struct _scv_tr_generator_default_data; |  | ||||||
| template <class T_begin, class T_end> class scv_tr_generator; |  | ||||||
|  |  | ||||||
| namespace iss { |  | ||||||
| class vm_if; |  | ||||||
| namespace arch { |  | ||||||
| template <typename BASE> class riscv_hart_m_p; |  | ||||||
| } |  | ||||||
| namespace debugger { |  | ||||||
| class target_adapter_if; |  | ||||||
| } |  | ||||||
| } // namespace iss |  | ||||||
|  |  | ||||||
| namespace sysc { | namespace sysc { | ||||||
|  |  | ||||||
| @@ -72,13 +62,12 @@ public: | |||||||
|  |  | ||||||
| namespace tgfs { | namespace tgfs { | ||||||
| class core_wrapper; | class core_wrapper; | ||||||
|  | struct core_trace; | ||||||
|  |  | ||||||
| class core_complex : public sc_core::sc_module, public scc::traceable { | class core_complex : public sc_core::sc_module, public scc::traceable { | ||||||
| public: | public: | ||||||
|     tlm::scc::initiator_mixin<tlm::scc::scv::tlm_rec_initiator_socket<32>> initiator{"intor"}; |     tlm::scc::initiator_mixin<tlm::scc::scv::tlm_rec_initiator_socket<32>> initiator{"intor"}; | ||||||
|  |  | ||||||
|     sc_core::sc_in<sc_core::sc_time> clk_i{"clk_i"}; |  | ||||||
|  |  | ||||||
|     sc_core::sc_in<bool> rst_i{"rst_i"}; |     sc_core::sc_in<bool> rst_i{"rst_i"}; | ||||||
|  |  | ||||||
|     sc_core::sc_in<bool> global_irq_i{"global_irq_i"}; |     sc_core::sc_in<bool> global_irq_i{"global_irq_i"}; | ||||||
| @@ -89,6 +78,9 @@ public: | |||||||
|  |  | ||||||
|     sc_core::sc_vector<sc_core::sc_in<bool>> local_irq_i{"local_irq_i", 16}; |     sc_core::sc_vector<sc_core::sc_in<bool>> local_irq_i{"local_irq_i", 16}; | ||||||
|  |  | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|  | 	sc_core::sc_in<sc_core::sc_time> clk_i{"clk_i"}; | ||||||
|  |  | ||||||
|     sc_core::sc_port<tlm::tlm_peek_if<uint64_t>, 1, sc_core::SC_ZERO_OR_MORE_BOUND> mtime_o; |     sc_core::sc_port<tlm::tlm_peek_if<uint64_t>, 1, sc_core::SC_ZERO_OR_MORE_BOUND> mtime_o; | ||||||
|  |  | ||||||
|     cci::cci_param<std::string> elf_file{"elf_file", ""}; |     cci::cci_param<std::string> elf_file{"elf_file", ""}; | ||||||
| @@ -97,6 +89,8 @@ public: | |||||||
|  |  | ||||||
|     cci::cci_param<uint64_t> reset_address{"reset_address", 0ULL}; |     cci::cci_param<uint64_t> reset_address{"reset_address", 0ULL}; | ||||||
|  |  | ||||||
|  |     cci::cci_param<std::string> core_type{"core_type", "tgc_c"}; | ||||||
|  |  | ||||||
|     cci::cci_param<std::string> backend{"backend", "interp"}; |     cci::cci_param<std::string> backend{"backend", "interp"}; | ||||||
|  |  | ||||||
|     cci::cci_param<unsigned short> gdb_server_port{"gdb_server_port", 0}; |     cci::cci_param<unsigned short> gdb_server_port{"gdb_server_port", 0}; | ||||||
| @@ -105,7 +99,47 @@ public: | |||||||
|  |  | ||||||
|     cci::cci_param<uint32_t> mhartid{"mhartid", 0}; |     cci::cci_param<uint32_t> mhartid{"mhartid", 0}; | ||||||
|  |  | ||||||
|     core_complex(sc_core::sc_module_name name); |     core_complex(sc_core::sc_module_name const& name); | ||||||
|  |  | ||||||
|  | #else | ||||||
|  | 	sc_core::sc_in<bool> clk_i{"clk_i"}; | ||||||
|  |  | ||||||
|  | 	sc_core::sc_in<uint64_t> mtime_i{"mtime_i"}; | ||||||
|  |  | ||||||
|  | 	scml_property<std::string> elf_file{"elf_file", ""}; | ||||||
|  |  | ||||||
|  |     scml_property<bool> enable_disass{"enable_disass", false}; | ||||||
|  |  | ||||||
|  |     scml_property<unsigned long long> reset_address{"reset_address", 0ULL}; | ||||||
|  |  | ||||||
|  |     scml_property<std::string> core_type{"core_type", "tgc_c"}; | ||||||
|  |  | ||||||
|  |     scml_property<std::string> backend{"backend", "interp"}; | ||||||
|  |  | ||||||
|  |     scml_property<unsigned> gdb_server_port{"gdb_server_port", 0}; | ||||||
|  |  | ||||||
|  |     scml_property<bool> dump_ir{"dump_ir", false}; | ||||||
|  |  | ||||||
|  |     scml_property<uint32_t> mhartid{"mhartid", 0}; | ||||||
|  |  | ||||||
|  |     core_complex(sc_core::sc_module_name const& name) | ||||||
|  |     : sc_module(name) | ||||||
|  |     , local_irq_i{"local_irq_i", 16} | ||||||
|  |     , elf_file{"elf_file", ""} | ||||||
|  |     , enable_disass{"enable_disass", false} | ||||||
|  |     , reset_address{"reset_address", 0ULL} | ||||||
|  |     , core_type{"core_type", "tgc_c"} | ||||||
|  |     , backend{"backend", "interp"} | ||||||
|  |     , gdb_server_port{"gdb_server_port", 0} | ||||||
|  |     , dump_ir{"dump_ir", false} | ||||||
|  |     , mhartid{"mhartid", 0} | ||||||
|  |     , read_lut(tlm_dmi_ext()) | ||||||
|  |     , write_lut(tlm_dmi_ext()) | ||||||
|  |     { | ||||||
|  |     	init(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | #endif | ||||||
|  |  | ||||||
|     ~core_complex(); |     ~core_complex(); | ||||||
|  |  | ||||||
| @@ -129,13 +163,14 @@ public: | |||||||
|  |  | ||||||
|     void trace(sc_core::sc_trace_file *trf) const override; |     void trace(sc_core::sc_trace_file *trf) const override; | ||||||
|  |  | ||||||
|     void disass_output(uint64_t pc, const std::string instr); |     bool disass_output(uint64_t pc, const std::string instr); | ||||||
|  |  | ||||||
|  |     void set_clock_period(sc_core::sc_time period); | ||||||
| protected: | protected: | ||||||
|     void before_end_of_elaboration() override; |     void before_end_of_elaboration() override; | ||||||
|     void start_of_simulation() override; |     void start_of_simulation() override; | ||||||
|  | 	void forward(); | ||||||
|     void run(); |     void run(); | ||||||
|     void clk_cb(); |  | ||||||
|     void rst_cb(); |     void rst_cb(); | ||||||
|     void sw_irq_cb(); |     void sw_irq_cb(); | ||||||
|     void timer_irq_cb(); |     void timer_irq_cb(); | ||||||
| @@ -144,23 +179,14 @@ protected: | |||||||
|     util::range_lut<tlm_dmi_ext> read_lut, write_lut; |     util::range_lut<tlm_dmi_ext> read_lut, write_lut; | ||||||
|     tlm_utils::tlm_quantumkeeper quantum_keeper; |     tlm_utils::tlm_quantumkeeper quantum_keeper; | ||||||
|     std::vector<uint8_t> write_buf; |     std::vector<uint8_t> write_buf; | ||||||
|     std::unique_ptr<core_wrapper> cpu; |     core_wrapper* cpu{nullptr}; | ||||||
|     std::unique_ptr<iss::vm_if> vm; |     sc_core::sc_signal<sc_core::sc_time> curr_clk; | ||||||
|     sc_core::sc_time curr_clk; |     core_trace* trc{nullptr}; | ||||||
|     iss::debugger::target_adapter_if *tgt_adapter; |     std::unique_ptr<scc::tick2time> t2t; | ||||||
| #ifdef WITH_SCV | private: | ||||||
|     //! transaction recording database |     void init(); | ||||||
|     scv_tr_db *m_db; |  | ||||||
|     //! blocking transaction recording stream handle |  | ||||||
|     scv_tr_stream *stream_handle; |  | ||||||
|     //! transaction generator handle for blocking transactions |  | ||||||
|     scv_tr_generator<_scv_tr_generator_default_data, _scv_tr_generator_default_data> *instr_tr_handle; |  | ||||||
|     scv_tr_generator<uint64_t, _scv_tr_generator_default_data> *fetch_tr_handle; |  | ||||||
|     scv_tr_handle tr_handle; |  | ||||||
| #endif |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| } /* namespace SiFive */ | } /* namespace SiFive */ | ||||||
| } /* namespace sysc */ | } /* namespace sysc */ | ||||||
|  |  | ||||||
| #endif /* _SYSC_SIFIVE_FE310_H_ */ | #endif /* _SYSC_CORE_COMPLEX_H_ */ | ||||||
|   | |||||||
| @@ -49,7 +49,9 @@ SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||||||
|  |  | ||||||
| /*---------------------------------------------------------------------------- | /*---------------------------------------------------------------------------- | ||||||
| *----------------------------------------------------------------------------*/ | *----------------------------------------------------------------------------*/ | ||||||
|  | #ifdef __GNUC__ | ||||||
| #define SOFTFLOAT_BUILTIN_CLZ 1 | #define SOFTFLOAT_BUILTIN_CLZ 1 | ||||||
| #define SOFTFLOAT_INTRINSIC_INT128 1 | #define SOFTFLOAT_INTRINSIC_INT128 1 | ||||||
|  | #endif | ||||||
| #include "opts-GCC.h" | #include "opts-GCC.h" | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								src/iss/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/iss/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1 @@ | |||||||
| /tgf_b.cpp | /tgc_*.cpp | ||||||
|   | |||||||
| @@ -32,26 +32,26 @@ | |||||||
| 
 | 
 | ||||||
| #include "util/ities.h" | #include "util/ities.h" | ||||||
| #include <util/logging.h> | #include <util/logging.h> | ||||||
| #include <iss/arch/tgf_c.h> | #include <iss/arch/tgc_c.h> | ||||||
| #include <cstdio> | #include <cstdio> | ||||||
| #include <cstring> | #include <cstring> | ||||||
| #include <fstream> | #include <fstream> | ||||||
| 
 | 
 | ||||||
| using namespace iss::arch; | using namespace iss::arch; | ||||||
| 
 | 
 | ||||||
| constexpr std::array<const char*, 35>    iss::arch::traits<iss::arch::tgf_c>::reg_names; | constexpr std::array<const char*, 35>    iss::arch::traits<iss::arch::tgc_c>::reg_names; | ||||||
| constexpr std::array<const char*, 35>    iss::arch::traits<iss::arch::tgf_c>::reg_aliases; | constexpr std::array<const char*, 35>    iss::arch::traits<iss::arch::tgc_c>::reg_aliases; | ||||||
| constexpr std::array<const uint32_t, 38> iss::arch::traits<iss::arch::tgf_c>::reg_bit_widths; | constexpr std::array<const uint32_t, 40> iss::arch::traits<iss::arch::tgc_c>::reg_bit_widths; | ||||||
| constexpr std::array<const uint32_t, 38> iss::arch::traits<iss::arch::tgf_c>::reg_byte_offsets; | constexpr std::array<const uint32_t, 40> iss::arch::traits<iss::arch::tgc_c>::reg_byte_offsets; | ||||||
| 
 | 
 | ||||||
| tgf_c::tgf_c() { | tgc_c::tgc_c() { | ||||||
|     reg.icount = 0; |     reg.icount = 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| tgf_c::~tgf_c() = default; | tgc_c::~tgc_c() = default; | ||||||
| 
 | 
 | ||||||
| void tgf_c::reset(uint64_t address) { | void tgc_c::reset(uint64_t address) { | ||||||
|     for(size_t i=0; i<traits<tgf_c>::NUM_REGS; ++i) set_reg(i, std::vector<uint8_t>(sizeof(traits<tgf_c>::reg_t),0)); |     for(size_t i=0; i<traits<tgc_c>::NUM_REGS; ++i) set_reg(i, std::vector<uint8_t>(sizeof(traits<tgc_c>::reg_t),0)); | ||||||
|     reg.PC=address; |     reg.PC=address; | ||||||
|     reg.NEXT_PC=reg.PC; |     reg.NEXT_PC=reg.PC; | ||||||
|     reg.PRIV=0x3; |     reg.PRIV=0x3; | ||||||
| @@ -59,11 +59,11 @@ void tgf_c::reset(uint64_t address) { | |||||||
|     reg.icount=0; |     reg.icount=0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| uint8_t *tgf_c::get_regs_base_ptr() { | uint8_t *tgc_c::get_regs_base_ptr() { | ||||||
| 	return reinterpret_cast<uint8_t*>(®); | 	return reinterpret_cast<uint8_t*>(®); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| tgf_c::phys_addr_t tgf_c::virt2phys(const iss::addr_t &pc) { | tgc_c::phys_addr_t tgc_c::virt2phys(const iss::addr_t &pc) { | ||||||
|     return phys_addr_t(pc); // change logical address to physical address
 |     return phys_addr_t(pc); // change logical address to physical address
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
							
								
								
									
										67
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										67
									
								
								src/main.cpp
									
									
									
									
									
								
							| @@ -31,15 +31,24 @@ | |||||||
|  *******************************************************************************/ |  *******************************************************************************/ | ||||||
|  |  | ||||||
| #include <iostream> | #include <iostream> | ||||||
| #include <iss/iss.h> | #include <iss/factory.h> | ||||||
|  |  | ||||||
| #include <boost/lexical_cast.hpp> | #include <boost/lexical_cast.hpp> | ||||||
| #include <boost/program_options.hpp> | #include <boost/program_options.hpp> | ||||||
| #include <iss/arch/riscv_hart_m_p.h> | #include <iss/arch/riscv_hart_m_p.h> | ||||||
| #ifdef WITH_TGF_B | #include "iss/arch/riscv_hart_m_p.h" | ||||||
| #include <iss/arch/tgf_b.h> | #include "iss/arch/tgc_c.h" | ||||||
|  | using tgc_c_plat_type = iss::arch::riscv_hart_m_p<iss::arch::tgc_c>; | ||||||
|  | #ifdef CORE_TGC_B | ||||||
|  | #include "iss/arch/riscv_hart_m_p.h" | ||||||
|  | #include "iss/arch/tgc_b.h" | ||||||
|  | using tgc_b_plat_type = iss::arch::riscv_hart_m_p<iss::arch::tgc_b>; | ||||||
|  | #endif | ||||||
|  | #ifdef CORE_TGC_D | ||||||
|  | #include "iss/arch/riscv_hart_mu_p.h" | ||||||
|  | #include "iss/arch/tgc_d.h" | ||||||
|  | using tgc_d_plat_type = iss::arch::riscv_hart_mu_p<iss::arch::tgc_d, (iss::arch::features_e)(iss::arch::FEAT_PMP | iss::arch::FEAT_CLIC | iss::arch::FEAT_EXT_N)>; | ||||||
| #endif | #endif | ||||||
| #include <iss/arch/tgf_c.h> |  | ||||||
| #ifdef WITH_LLVM | #ifdef WITH_LLVM | ||||||
| #include <iss/llvm/jit_helper.h> | #include <iss/llvm/jit_helper.h> | ||||||
| #endif | #endif | ||||||
| @@ -49,23 +58,6 @@ | |||||||
|  |  | ||||||
| namespace po = boost::program_options; | namespace po = boost::program_options; | ||||||
|  |  | ||||||
| using cpu_ptr = std::unique_ptr<iss::arch_if>; |  | ||||||
| using vm_ptr= std::unique_ptr<iss::vm_if>; |  | ||||||
|  |  | ||||||
| template<typename CORE> |  | ||||||
| std::tuple<cpu_ptr, vm_ptr> create_cpu(std::string const& backend, unsigned gdb_port){ |  | ||||||
|     CORE* lcpu = new iss::arch::riscv_hart_m_p<CORE>(); |  | ||||||
|     if(backend == "interp") |  | ||||||
|         return {cpu_ptr{lcpu}, vm_ptr{iss::interp::create(lcpu, gdb_port)}}; |  | ||||||
| #ifdef WITH_LLVM |  | ||||||
|     if(backend == "llvm") |  | ||||||
|         return {cpu_ptr{lcpu}, vm_ptr{iss::llvm::create(lcpu, gdb_port)}}; |  | ||||||
| #endif |  | ||||||
| //    if(backend == "tcc") |  | ||||||
| //        return {cpu_ptr{lcpu}, vm_ptr{iss::tcc::create(lcpu, gdb_port)}}; |  | ||||||
|     return {nullptr, nullptr}; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| int main(int argc, char *argv[]) { | int main(int argc, char *argv[]) { | ||||||
|     /* |     /* | ||||||
|      *  Define and parse the program options |      *  Define and parse the program options | ||||||
| @@ -86,7 +78,7 @@ int main(int argc, char *argv[]) { | |||||||
|         ("mem,m", po::value<std::string>(), "the memory input file") |         ("mem,m", po::value<std::string>(), "the memory input file") | ||||||
|         ("plugin,p", po::value<std::vector<std::string>>(), "plugin to activate") |         ("plugin,p", po::value<std::vector<std::string>>(), "plugin to activate") | ||||||
|         ("backend", po::value<std::string>()->default_value("interp"), "the memory input file") |         ("backend", po::value<std::string>()->default_value("interp"), "the memory input file") | ||||||
|         ("isa", po::value<std::string>()->default_value("tgf_c"), "isa to use for simulation"); |         ("isa", po::value<std::string>()->default_value("tgc_c"), "isa to use for simulation"); | ||||||
|     // clang-format on |     // clang-format on | ||||||
|     auto parsed = po::command_line_parser(argc, argv).options(desc).allow_unregistered().run(); |     auto parsed = po::command_line_parser(argc, argv).options(desc).allow_unregistered().run(); | ||||||
|     try { |     try { | ||||||
| @@ -128,20 +120,27 @@ int main(int argc, char *argv[]) { | |||||||
| #endif | #endif | ||||||
|         bool dump = clim.count("dump-ir"); |         bool dump = clim.count("dump-ir"); | ||||||
|         // instantiate the simulator |         // instantiate the simulator | ||||||
|         vm_ptr vm{nullptr}; |         iss::vm_ptr vm{nullptr}; | ||||||
|         cpu_ptr cpu{nullptr}; |         iss::cpu_ptr cpu{nullptr}; | ||||||
|         std::string isa_opt(clim["isa"].as<std::string>()); |         std::string isa_opt(clim["isa"].as<std::string>()); | ||||||
| #ifdef WITH_TGF_B |         if (isa_opt == "tgc_c") { | ||||||
|         if (isa_opt == "tgf_b") { |  | ||||||
|             std::tie(cpu, vm) = |             std::tie(cpu, vm) = | ||||||
|                 create_cpu<iss::arch::tgf_b>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); |                 iss::create_cpu<tgc_c_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||||
|  |         } else | ||||||
|  | #ifdef CORE_TGC_B | ||||||
|  |         if (isa_opt == "tgc_b") { | ||||||
|  |             std::tie(cpu, vm) = | ||||||
|  |                 iss::create_cpu<tgc_b_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||||
|         } else |         } else | ||||||
| #endif | #endif | ||||||
|             if (isa_opt == "tgf_c") { | #ifdef CORE_TGC_D | ||||||
|  |         if (isa_opt == "tgc_d") { | ||||||
|             std::tie(cpu, vm) = |             std::tie(cpu, vm) = | ||||||
|                 create_cpu<iss::arch::tgf_c>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); |                 iss::create_cpu<tgc_d_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||||
|         } else { |         } else | ||||||
|             LOG(ERROR) << "Illegal argument value for '--isa': " << clim["isa"].as<std::string>() << std::endl; | #endif | ||||||
|  |         { | ||||||
|  |             LOG(ERR) << "Illegal argument value for '--isa': " << clim["isa"].as<std::string>() << std::endl; | ||||||
|             return 127; |             return 127; | ||||||
|         } |         } | ||||||
|         if (clim.count("plugin")) { |         if (clim.count("plugin")) { | ||||||
| @@ -162,7 +161,7 @@ int main(int argc, char *argv[]) { | |||||||
|                     vm->register_plugin(*ce_plugin); |                     vm->register_plugin(*ce_plugin); | ||||||
|                     plugin_list.push_back(ce_plugin); |                     plugin_list.push_back(ce_plugin); | ||||||
|                 } else { |                 } else { | ||||||
|                     LOG(ERROR) << "Unknown plugin name: " << plugin_name << ", valid names are 'ce', 'ic'" << std::endl; |                     LOG(ERR) << "Unknown plugin name: " << plugin_name << ", valid names are 'ce', 'ic'" << std::endl; | ||||||
|                     return 127; |                     return 127; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| @@ -179,7 +178,7 @@ int main(int argc, char *argv[]) { | |||||||
|         } |         } | ||||||
|         uint64_t start_address = 0; |         uint64_t start_address = 0; | ||||||
|         if (clim.count("mem")) |         if (clim.count("mem")) | ||||||
|             vm->get_arch()->load_file(clim["mem"].as<std::string>(), iss::arch::traits<iss::arch::tgf_c>::MEM); |             vm->get_arch()->load_file(clim["mem"].as<std::string>()); | ||||||
|         if (clim.count("elf")) |         if (clim.count("elf")) | ||||||
|             for (std::string input : clim["elf"].as<std::vector<std::string>>()) { |             for (std::string input : clim["elf"].as<std::vector<std::string>>()) { | ||||||
|                 auto start_addr = vm->get_arch()->load_file(input); |                 auto start_addr = vm->get_arch()->load_file(input); | ||||||
| @@ -197,7 +196,7 @@ int main(int argc, char *argv[]) { | |||||||
|         auto cycles = clim["instructions"].as<uint64_t>(); |         auto cycles = clim["instructions"].as<uint64_t>(); | ||||||
|         res = vm->start(cycles, dump); |         res = vm->start(cycles, dump); | ||||||
|     } catch (std::exception &e) { |     } catch (std::exception &e) { | ||||||
|         LOG(ERROR) << "Unhandled Exception reached the top of main: " << e.what() << ", application will now exit" |         LOG(ERR) << "Unhandled Exception reached the top of main: " << e.what() << ", application will now exit" | ||||||
|                    << std::endl; |                    << std::endl; | ||||||
|         res = 2; |         res = 2; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -47,10 +47,10 @@ iss::plugin::cycle_estimate::cycle_estimate(std::string config_file_name) | |||||||
|             try { |             try { | ||||||
|                 is >> root; |                 is >> root; | ||||||
|             } catch (Json::RuntimeError &e) { |             } catch (Json::RuntimeError &e) { | ||||||
|                 LOG(ERROR) << "Could not parse input file " << config_file_name << ", reason: " << e.what(); |                 LOG(ERR) << "Could not parse input file " << config_file_name << ", reason: " << e.what(); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             LOG(ERROR) << "Could not open input file " << config_file_name; |             LOG(ERR) << "Could not open input file " << config_file_name; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -77,7 +77,7 @@ bool iss::plugin::cycle_estimate::registration(const char* const version, vm_if& | |||||||
|     		} |     		} | ||||||
|     	} |     	} | ||||||
|     } else { |     } else { | ||||||
|         LOG(ERROR)<<"plugin cycle_estimate: could not find an entry for "<<core_name<<" in JSON file"<<std::endl; |         LOG(ERR)<<"plugin cycle_estimate: could not find an entry for "<<core_name<<" in JSON file"<<std::endl; | ||||||
|     } |     } | ||||||
| 	return true; | 	return true; | ||||||
|  |  | ||||||
| @@ -87,6 +87,8 @@ void iss::plugin::cycle_estimate::callback(instr_info_t instr_info, exec_info co | |||||||
|     assert(arch_instr && "No instrumentation interface available but callback executed"); |     assert(arch_instr && "No instrumentation interface available but callback executed"); | ||||||
| 	auto entry = delays[instr_info.instr_id]; | 	auto entry = delays[instr_info.instr_id]; | ||||||
| 	bool taken = (arch_instr->get_next_pc()-arch_instr->get_pc()) != (entry.size/8); | 	bool taken = (arch_instr->get_next_pc()-arch_instr->get_pc()) != (entry.size/8); | ||||||
|     uint32_t delay = taken ? entry.taken : entry.not_taken; |     if (taken && entry.taken > 1) | ||||||
|     if(delay>1) arch_instr->set_curr_instr_cycles(delay); |         arch_instr->set_curr_instr_cycles(entry.taken); | ||||||
|  |     else if (entry.not_taken > 1) | ||||||
|  |         arch_instr->set_curr_instr_cycles(entry.not_taken); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -46,10 +46,10 @@ iss::plugin::instruction_count::instruction_count(std::string config_file_name) | |||||||
|             try { |             try { | ||||||
|                 is >> root; |                 is >> root; | ||||||
|             } catch (Json::RuntimeError &e) { |             } catch (Json::RuntimeError &e) { | ||||||
|                 LOG(ERROR) << "Could not parse input file " << config_file_name << ", reason: " << e.what(); |                 LOG(ERR) << "Could not parse input file " << config_file_name << ", reason: " << e.what(); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             LOG(ERROR) << "Could not open input file " << config_file_name; |             LOG(ERR) << "Could not open input file " << config_file_name; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -85,7 +85,7 @@ bool iss::plugin::instruction_count::registration(const char* const version, vm_ | |||||||
|     	} |     	} | ||||||
|     	rep_counts.resize(delays.size()); |     	rep_counts.resize(delays.size()); | ||||||
|     } else { |     } else { | ||||||
|         LOG(ERROR)<<"plugin instruction_count: could not find an entry for "<<core_name<<" in JSON file"<<std::endl; |         LOG(ERR)<<"plugin instruction_count: could not find an entry for "<<core_name<<" in JSON file"<<std::endl; | ||||||
|     } |     } | ||||||
| 	return true; | 	return true; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -30,22 +30,54 @@ | |||||||
|  * |  * | ||||||
|  *******************************************************************************/ |  *******************************************************************************/ | ||||||
|  |  | ||||||
| #include "sysc/core_complex.h" | // clang-format off | ||||||
| #include "iss/arch/riscv_hart_m_p.h" |  | ||||||
| #include "iss/arch/tgf_c.h" |  | ||||||
| #include "iss/debugger/encoderdecoder.h" |  | ||||||
| #include "iss/debugger/gdb_session.h" | #include "iss/debugger/gdb_session.h" | ||||||
|  | #include "iss/debugger/encoderdecoder.h" | ||||||
| #include "iss/debugger/server.h" | #include "iss/debugger/server.h" | ||||||
| #include "iss/debugger/target_adapter_if.h" | #include "iss/debugger/target_adapter_if.h" | ||||||
| #include "iss/iss.h" | #include "iss/iss.h" | ||||||
| #include "iss/vm_types.h" | #include "iss/vm_types.h" | ||||||
|  | #include "sysc/core_complex.h" | ||||||
|  | #ifdef CORE_TGC_B | ||||||
|  | #include "iss/arch/riscv_hart_m_p.h" | ||||||
|  | #include "iss/arch/tgc_b.h" | ||||||
|  | using tgc_b_plat_type = iss::arch::riscv_hart_m_p<iss::arch::tgc_b>; | ||||||
|  | #endif | ||||||
|  | #include "iss/arch/riscv_hart_m_p.h" | ||||||
|  | #include "iss/arch/tgc_c.h" | ||||||
|  | using tgc_c_plat_type = iss::arch::riscv_hart_m_p<iss::arch::tgc_c>; | ||||||
|  | #ifdef CORE_TGC_D | ||||||
|  | #include "iss/arch/riscv_hart_mu_p.h" | ||||||
|  | #include "iss/arch/tgc_d.h" | ||||||
|  | using tgc_d_plat_type = iss::arch::riscv_hart_mu_p<iss::arch::tgc_d, iss::arch::FEAT_PMP>; | ||||||
|  | #endif | ||||||
| #include "scc/report.h" | #include "scc/report.h" | ||||||
| #include <iostream> | #include <iostream> | ||||||
| #include <sstream> | #include <sstream> | ||||||
|  | #include <array> | ||||||
|  | // clang-format on | ||||||
|  |  | ||||||
|  | #define STR(X) #X | ||||||
|  | #define CREATE_CORE(CN) \ | ||||||
|  | if (type == STR(CN)) { std::tie(cpu, vm) = create_core<CN ## _plat_type>(backend, gdb_port, hart_id); } else | ||||||
|  |  | ||||||
| #ifdef WITH_SCV | #ifdef WITH_SCV | ||||||
| #include <array> |  | ||||||
| #include <scv.h> | #include <scv.h> | ||||||
|  | #else | ||||||
|  | #include <scv-tr.h> | ||||||
|  | using namespace scv_tr; | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|  | #define GET_PROP_VALUE(P) P.get_value() | ||||||
|  | #else | ||||||
|  | #define GET_PROP_VALUE(P) P.getValue() | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef _MSC_VER | ||||||
|  | // not #if defined(_WIN32) || defined(_WIN64) because we have strncasecmp in mingw | ||||||
|  | #define strncasecmp _strnicmp | ||||||
|  | #define strcasecmp _stricmp | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| namespace sysc { | namespace sysc { | ||||||
| @@ -57,43 +89,17 @@ using namespace sc_core; | |||||||
|  |  | ||||||
| namespace { | namespace { | ||||||
| iss::debugger::encoder_decoder encdec; | iss::debugger::encoder_decoder encdec; | ||||||
| } |  | ||||||
|  |  | ||||||
| using core_type = iss::arch::tgf_c; |  | ||||||
|  |  | ||||||
| namespace { |  | ||||||
|  |  | ||||||
| std::array<const char, 4> lvl = {{'U', 'S', 'H', 'M'}}; | std::array<const char, 4> lvl = {{'U', 'S', 'H', 'M'}}; | ||||||
|  |  | ||||||
| std::array<const char*, 16> trap_str = { { |  | ||||||
| 		"Instruction address misaligned", |  | ||||||
| 		"Instruction access fault", |  | ||||||
| 		"Illegal instruction", |  | ||||||
| 		"Breakpoint", |  | ||||||
| 		"Load address misaligned", |  | ||||||
| 		"Load access fault", |  | ||||||
| 		"Store/AMO address misaligned", |  | ||||||
| 		"Store/AMO access fault", |  | ||||||
| 		"Environment call from U-mode", |  | ||||||
| 		"Environment call from S-mode", |  | ||||||
| 		"Reserved", |  | ||||||
| 		"Environment call from M-mode", |  | ||||||
| 		"Instruction page fault", |  | ||||||
| 		"Load page fault", |  | ||||||
| 		"Reserved", |  | ||||||
| 		"Store/AMO page fault" |  | ||||||
| } }; |  | ||||||
| std::array<const char*, 12> irq_str = { { |  | ||||||
| 		"User software interrupt", "Supervisor software interrupt", "Reserved", "Machine software interrupt", |  | ||||||
| 		"User timer interrupt",    "Supervisor timer interrupt",    "Reserved", "Machine timer interrupt", |  | ||||||
| 		"User external interrupt", "Supervisor external interrupt", "Reserved", "Machine external interrupt" } }; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| class core_wrapper : public iss::arch::riscv_hart_m_p<core_type> { | template<typename PLAT> | ||||||
|  | class core_wrapper_t : public PLAT { | ||||||
| public: | public: | ||||||
|     using base_type = arch::riscv_hart_m_p<core_type>; |     using reg_t       = typename arch::traits<typename PLAT::core>::reg_t; | ||||||
|     using phys_addr_t = typename arch::traits<core_type>::phys_addr_t; |     using phys_addr_t = typename arch::traits<typename PLAT::core>::phys_addr_t; | ||||||
|     core_wrapper(core_complex *owner) |     using heart_state_t = typename PLAT::hart_state_type; | ||||||
|  |     core_wrapper_t(core_complex *owner) | ||||||
|     : owner(owner) { } |     : owner(owner) { } | ||||||
|  |  | ||||||
|     uint32_t get_mode() { return this->reg.PRIV; } |     uint32_t get_mode() { return this->reg.PRIV; } | ||||||
| @@ -102,24 +108,23 @@ public: | |||||||
|  |  | ||||||
|     inline bool get_interrupt_execution() { return this->interrupt_sim; } |     inline bool get_interrupt_execution() { return this->interrupt_sim; } | ||||||
|  |  | ||||||
|     base_type::hart_state<base_type::reg_t> &get_state() { return this->state; } |     heart_state_t &get_state() { return this->state; } | ||||||
|  |  | ||||||
|     void notify_phase(exec_phase p) override { |     void notify_phase(iss::arch_if::exec_phase p) override { | ||||||
|         if (p == ISTART) owner->sync(this->reg.icount + cycle_offset); |         if (p == iss::arch_if::ISTART) owner->sync(this->reg.icount); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     sync_type needed_sync() const override { return PRE_SYNC; } |     sync_type needed_sync() const override { return PRE_SYNC; } | ||||||
|  |  | ||||||
|     void disass_output(uint64_t pc, const std::string instr) override { |     void disass_output(uint64_t pc, const std::string instr) override { | ||||||
|         if (INFO <= Log<Output2FILE<disass>>::reporting_level() && Output2FILE<disass>::stream()) { |         if (!owner->disass_output(pc, instr)) { | ||||||
|             std::stringstream s; |             std::stringstream s; | ||||||
|             s << "[p:" << lvl[this->reg.PRIV] << ";s:0x" << std::hex << std::setfill('0') |             s << "[p:" << lvl[this->reg.PRIV] << ";s:0x" << std::hex << std::setfill('0') | ||||||
|               << std::setw(sizeof(reg_t) * 2) << (reg_t)state.mstatus << std::dec << ";c:" << this->reg.icount << "]"; |               << std::setw(sizeof(reg_t) * 2) << (reg_t)this->state.mstatus << std::dec << ";c:" << this->reg.icount << "]"; | ||||||
|             Log<Output2FILE<disass>>().get(INFO, "disass") |             SCCDEBUG(owner->name())<<"disass: " | ||||||
|                 << "0x" << std::setw(16) << std::right << std::setfill('0') << std::hex << pc << "\t\t" << std::setw(40) |                 << "0x" << std::setw(16) << std::right << std::setfill('0') << std::hex << pc << "\t\t" << std::setw(40) | ||||||
|                 << std::setfill(' ') << std::left << instr << s.str(); |                 << std::setfill(' ') << std::left << instr << s.str(); | ||||||
|         } |         } | ||||||
|         owner->disass_output(pc, instr); |  | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     status read_mem(phys_addr_t addr, unsigned length, uint8_t *const data) override { |     status read_mem(phys_addr_t addr, unsigned length, uint8_t *const data) override { | ||||||
| @@ -146,6 +151,7 @@ public: | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     status read_csr(unsigned addr, reg_t &val) override { |     status read_csr(unsigned addr, reg_t &val) override { | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|         if((addr==arch::time || addr==arch::timeh) && owner->mtime_o.get_interface(0)){ |         if((addr==arch::time || addr==arch::timeh) && owner->mtime_o.get_interface(0)){ | ||||||
|             uint64_t time_val; |             uint64_t time_val; | ||||||
|             bool ret = owner->mtime_o->nb_peek(time_val); |             bool ret = owner->mtime_o->nb_peek(time_val); | ||||||
| @@ -156,21 +162,32 @@ public: | |||||||
|                 val = static_cast<reg_t>(time_val >> 32); |                 val = static_cast<reg_t>(time_val >> 32); | ||||||
|             } |             } | ||||||
|             return ret?Ok:Err; |             return ret?Ok:Err; | ||||||
|  | #else | ||||||
|  | 		if((addr==arch::time || addr==arch::timeh)){ | ||||||
|  | 			uint64_t time_val = owner->mtime_i.read(); | ||||||
|  | 			if (addr == iss::arch::time) { | ||||||
|  | 				val = static_cast<reg_t>(time_val); | ||||||
|  | 			} else if (addr == iss::arch::timeh) { | ||||||
|  | 				if (sizeof(reg_t) != 4) return iss::Err; | ||||||
|  | 				val = static_cast<reg_t>(time_val >> 32); | ||||||
|  | 			} | ||||||
|  | 			return Ok; | ||||||
|  | #endif | ||||||
|         } else { |         } else { | ||||||
|             return base_type::read_csr(addr, val); |             return PLAT::read_csr(addr, val); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void wait_until(uint64_t flags) override { |     void wait_until(uint64_t flags) override { | ||||||
|         SCCDEBUG(owner->name()) << "Sleeping until interrupt"; |         SCCDEBUG(owner->name()) << "Sleeping until interrupt"; | ||||||
|         do { |         do { | ||||||
|             wait(wfi_evt); |             sc_core::wait(wfi_evt); | ||||||
|         } while (this->reg.pending_trap == 0); |         } while (this->reg.pending_trap == 0); | ||||||
|         base_type::wait_until(flags); |         PLAT::wait_until(flags); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     void local_irq(short id, bool value) { |     void local_irq(short id, bool value) { | ||||||
|         base_type::reg_t mask = 0; |         reg_t mask = 0; | ||||||
|         switch (id) { |         switch (id) { | ||||||
|         case 16: // SW |         case 16: // SW | ||||||
|             mask = 1 << 3; |             mask = 1 << 3; | ||||||
| @@ -230,19 +247,95 @@ int cmd_sysc(int argc, char *argv[], debugger::out_func of, debugger::data_func | |||||||
|     return Err; |     return Err; | ||||||
| } | } | ||||||
|  |  | ||||||
| core_complex::core_complex(sc_module_name name) | using cpu_ptr = std::unique_ptr<iss::arch_if>; | ||||||
|  | using vm_ptr= std::unique_ptr<iss::vm_if>; | ||||||
|  |  | ||||||
|  | class core_wrapper { | ||||||
|  | public: | ||||||
|  |     core_wrapper(core_complex *owner) : owner(owner) { } | ||||||
|  |  | ||||||
|  |     void reset(uint64_t addr){vm->reset(addr);} | ||||||
|  |     inline void start(){vm->start();} | ||||||
|  |     inline std::pair<uint64_t, bool> load_file(std::string const& name){ return cpu->load_file(name);}; | ||||||
|  |  | ||||||
|  |     std::function<unsigned(void)> get_mode; | ||||||
|  |     std::function<uint64_t(void)> get_state; | ||||||
|  |     std::function<bool(void)> get_interrupt_execution; | ||||||
|  |     std::function<void(bool)> set_interrupt_execution; | ||||||
|  |     std::function<void(short, bool)> local_irq; | ||||||
|  |  | ||||||
|  |     template<typename PLAT> | ||||||
|  |     std::tuple<cpu_ptr, vm_ptr> create_core(std::string const& backend, unsigned gdb_port, uint32_t hart_id){ | ||||||
|  |         auto* lcpu = new core_wrapper_t<PLAT>(owner); | ||||||
|  |         lcpu->set_mhartid(hart_id); | ||||||
|  |         get_mode = [lcpu]() { return lcpu->get_mode(); }; | ||||||
|  |         get_state = [lcpu]() { return lcpu->get_state().mstatus.backing.val; }; | ||||||
|  |         get_interrupt_execution = [lcpu]() { return lcpu->get_interrupt_execution(); }; | ||||||
|  |         set_interrupt_execution = [lcpu](bool b) { return lcpu->set_interrupt_execution(b); }; | ||||||
|  |         local_irq = [lcpu](short s, bool b) { return lcpu->local_irq(s, b); }; | ||||||
|  |         if(backend == "interp") | ||||||
|  |             return {cpu_ptr{lcpu}, vm_ptr{iss::interp::create(static_cast<typename PLAT::core*>(lcpu), gdb_port)}}; | ||||||
|  | #ifdef WITH_LLVM | ||||||
|  |         if(backend == "llvm") | ||||||
|  |             return {cpu_ptr{lcpu}, vm_ptr{iss::llvm::create(lcpu, gdb_port)}}; | ||||||
|  | #endif | ||||||
|  | #ifdef WITH_TCC | ||||||
|  |         if(backend == "tcc") | ||||||
|  |     s        return {cpu_ptr{lcpu}, vm_ptr{iss::tcc::create(lcpu, gdb_port)}}; | ||||||
|  | #endif | ||||||
|  |         return {nullptr, nullptr}; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void create_cpu(std::string const& type, std::string const& backend, unsigned gdb_port, uint32_t hart_id){ | ||||||
|  |         CREATE_CORE(tgc_c) | ||||||
|  | #ifdef CORE_TGC_B | ||||||
|  |         CREATE_CORE(tgc_b) | ||||||
|  | #endif | ||||||
|  | #ifdef CORE_TGC_D | ||||||
|  |         CREATE_CORE(tgc_d) | ||||||
|  | #endif | ||||||
|  |         { | ||||||
|  |             LOG(ERR) << "Illegal argument value for core type: " << type << std::endl; | ||||||
|  |         } | ||||||
|  |         auto *srv = debugger::server<debugger::gdb_session>::get(); | ||||||
|  |         if (srv) tgt_adapter = srv->get_target(); | ||||||
|  |         if (tgt_adapter) | ||||||
|  |             tgt_adapter->add_custom_command( | ||||||
|  |                 {"sysc", [this](int argc, char *argv[], debugger::out_func of, | ||||||
|  |                                 debugger::data_func df) -> int { return cmd_sysc(argc, argv, of, df, tgt_adapter); }, | ||||||
|  |                  "SystemC sub-commands: break <time>, print_time"}); | ||||||
|  |  | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     core_complex * const owner; | ||||||
|  |     vm_ptr vm{nullptr}; | ||||||
|  |     cpu_ptr cpu{nullptr}; | ||||||
|  |     iss::debugger::target_adapter_if *tgt_adapter{nullptr}; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct core_trace { | ||||||
|  |     //! transaction recording database | ||||||
|  |     scv_tr_db *m_db{nullptr}; | ||||||
|  |     //! blocking transaction recording stream handle | ||||||
|  |     scv_tr_stream *stream_handle{nullptr}; | ||||||
|  |     //! transaction generator handle for blocking transactions | ||||||
|  |     scv_tr_generator<_scv_tr_generator_default_data, _scv_tr_generator_default_data> *instr_tr_handle{nullptr}; | ||||||
|  |     scv_tr_handle tr_handle; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | SC_HAS_PROCESS(core_complex);// NOLINT | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|  | core_complex::core_complex(sc_module_name const& name) | ||||||
| : sc_module(name) | : sc_module(name) | ||||||
| , read_lut(tlm_dmi_ext()) | , read_lut(tlm_dmi_ext()) | ||||||
| , write_lut(tlm_dmi_ext()) | , write_lut(tlm_dmi_ext()) | ||||||
| , tgt_adapter(nullptr) |  | ||||||
| #ifdef WITH_SCV |  | ||||||
| , m_db(scv_tr_db::get_default_db()) |  | ||||||
| , stream_handle(nullptr) |  | ||||||
| , instr_tr_handle(nullptr) |  | ||||||
| , fetch_tr_handle(nullptr) |  | ||||||
| #endif |  | ||||||
| { | { | ||||||
|     SC_HAS_PROCESS(core_complex);// NOLINT | 	init(); | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | void core_complex::init(){ | ||||||
|  | 	trc=new core_trace(); | ||||||
|     initiator.register_invalidate_direct_mem_ptr([=](uint64_t start, uint64_t end) -> void { |     initiator.register_invalidate_direct_mem_ptr([=](uint64_t start, uint64_t end) -> void { | ||||||
|         auto lut_entry = read_lut.getEntry(start); |         auto lut_entry = read_lut.getEntry(start); | ||||||
|         if (lut_entry.get_granted_access() != tlm::tlm_dmi::DMI_ACCESS_NONE && end <= lut_entry.get_end_address() + 1) { |         if (lut_entry.get_granted_access() != tlm::tlm_dmi::DMI_ACCESS_NONE && end <= lut_entry.get_end_address() + 1) { | ||||||
| @@ -255,8 +348,6 @@ core_complex::core_complex(sc_module_name name) | |||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     SC_THREAD(run); |     SC_THREAD(run); | ||||||
|     SC_METHOD(clk_cb); |  | ||||||
|     sensitive << clk_i; |  | ||||||
|     SC_METHOD(rst_cb); |     SC_METHOD(rst_cb); | ||||||
|     sensitive << rst_i; |     sensitive << rst_i; | ||||||
|     SC_METHOD(sw_irq_cb); |     SC_METHOD(sw_irq_cb); | ||||||
| @@ -265,85 +356,82 @@ core_complex::core_complex(sc_module_name name) | |||||||
|     sensitive << timer_irq_i; |     sensitive << timer_irq_i; | ||||||
|     SC_METHOD(global_irq_cb); |     SC_METHOD(global_irq_cb); | ||||||
|     sensitive << global_irq_i; |     sensitive << global_irq_i; | ||||||
|  |     trc->m_db=scv_tr_db::get_default_db(); | ||||||
|  |  | ||||||
|  | 	SC_METHOD(forward); | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|  | 	sensitive<<clk_i; | ||||||
|  | #else | ||||||
|  | 	sensitive<<curr_clk; | ||||||
|  | 	t2t.reset(new scc::tick2time{"t2t"}); | ||||||
|  | 	t2t->clk_i(clk_i); | ||||||
|  | 	t2t->clk_o(curr_clk); | ||||||
|  | #endif | ||||||
| } | } | ||||||
|  |  | ||||||
| core_complex::~core_complex() = default; | core_complex::~core_complex(){ | ||||||
|  |     delete cpu; | ||||||
|  |     delete trc; | ||||||
|  | } | ||||||
|  |  | ||||||
| void core_complex::trace(sc_trace_file *trf) const {} | void core_complex::trace(sc_trace_file *trf) const {} | ||||||
|  |  | ||||||
| using vm_ptr= std::unique_ptr<iss::vm_if>; |  | ||||||
| vm_ptr create_cpu(core_wrapper* cpu, std::string const& backend, unsigned gdb_port){ |  | ||||||
|     if(backend == "interp") |  | ||||||
|         return vm_ptr{iss::interp::create<core_type>(cpu, gdb_port)}; |  | ||||||
| #ifdef WITH_LLVM |  | ||||||
|     if(backend == "llvm") |  | ||||||
|         return vm_ptr{iss::llvm::create(lcpu, gdb_port)}; |  | ||||||
| #endif |  | ||||||
| #ifdef WITH_TCC |  | ||||||
|     if(backend == "tcc") |  | ||||||
|         return vm_ptr{iss::tcc::create<core_type>(cpu, gdb_port)}; |  | ||||||
| #endif |  | ||||||
|     return {nullptr}; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void core_complex::before_end_of_elaboration() { | void core_complex::before_end_of_elaboration() { | ||||||
|     SCCDEBUG(SCMOD)<<"instantiating iss::arch::tgf with "<<backend.get_value()<<" backend"; |     SCCDEBUG(SCMOD)<<"instantiating iss::arch::tgf with "<<GET_PROP_VALUE(backend)<<" backend"; | ||||||
|     cpu = scc::make_unique<core_wrapper>(this); |     // cpu = scc::make_unique<core_wrapper>(this); | ||||||
|     cpu->set_mhartid(mhartid.get_value()); |     cpu = new core_wrapper(this); | ||||||
|  |     cpu->create_cpu(GET_PROP_VALUE(core_type), GET_PROP_VALUE(backend), GET_PROP_VALUE(gdb_server_port), GET_PROP_VALUE(mhartid)); | ||||||
|     vm = create_cpu(cpu.get(), backend.get_value(), gdb_server_port.get_value()); |     sc_assert(cpu->vm!=nullptr); | ||||||
|     sc_assert(vm!=nullptr); |     cpu->vm->setDisassEnabled(GET_PROP_VALUE(enable_disass) || trc->m_db != nullptr); | ||||||
| #ifdef WITH_SCV |  | ||||||
|     vm->setDisassEnabled(enable_disass.get_value() || m_db != nullptr); |  | ||||||
| #else |  | ||||||
|     vm->setDisassEnabled(enable_disass.get_value()); |  | ||||||
| #endif |  | ||||||
|     auto *srv = debugger::server<debugger::gdb_session>::get(); |  | ||||||
|     if (srv) tgt_adapter = srv->get_target(); |  | ||||||
|     if (tgt_adapter) |  | ||||||
|         tgt_adapter->add_custom_command( |  | ||||||
|             {"sysc", [this](int argc, char *argv[], debugger::out_func of, |  | ||||||
|                             debugger::data_func df) -> int { return cmd_sysc(argc, argv, of, df, tgt_adapter); }, |  | ||||||
|              "SystemC sub-commands: break <time>, print_time"}); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void core_complex::start_of_simulation() { | void core_complex::start_of_simulation() { | ||||||
|     quantum_keeper.reset(); |     quantum_keeper.reset(); | ||||||
|     if (elf_file.get_value().size() > 0) { |     if (GET_PROP_VALUE(elf_file).size() > 0) { | ||||||
|         istringstream is(elf_file.get_value()); |         istringstream is(GET_PROP_VALUE(elf_file)); | ||||||
|         string s; |         string s; | ||||||
|         while (getline(is, s, ',')) { |         while (getline(is, s, ',')) { | ||||||
|             std::pair<uint64_t, bool> start_addr = cpu->load_file(s); |             std::pair<uint64_t, bool> start_addr = cpu->load_file(s); | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|             if (reset_address.is_default_value() && start_addr.second == true) |             if (reset_address.is_default_value() && start_addr.second == true) | ||||||
|                 reset_address.set_value(start_addr.first); |                 reset_address.set_value(start_addr.first); | ||||||
|  | #else | ||||||
|  |             if (start_addr.second == true) | ||||||
|  |                 reset_address=start_addr.first; | ||||||
|  | #endif | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| #ifdef WITH_SCV |     if (trc->m_db != nullptr && trc->stream_handle == nullptr) { | ||||||
|     if (m_db != nullptr && stream_handle == nullptr) { |  | ||||||
|         string basename(this->name()); |         string basename(this->name()); | ||||||
|         stream_handle = new scv_tr_stream((basename + ".instr").c_str(), "TRANSACTOR", m_db); |         trc->stream_handle = new scv_tr_stream((basename + ".instr").c_str(), "TRANSACTOR", trc->m_db); | ||||||
|         instr_tr_handle = new scv_tr_generator<>("execute", *stream_handle); |         trc->instr_tr_handle = new scv_tr_generator<>("execute", *trc->stream_handle); | ||||||
|         fetch_tr_handle = new scv_tr_generator<uint64_t>("fetch", *stream_handle); |  | ||||||
|     } |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool core_complex::disass_output(uint64_t pc, const std::string instr_str) { | ||||||
|  |     if (trc->m_db == nullptr) return false; | ||||||
|  |     if (trc->tr_handle.is_active()) trc->tr_handle.end_transaction(); | ||||||
|  |     trc->tr_handle = trc->instr_tr_handle->begin_transaction(); | ||||||
|  |     trc->tr_handle.record_attribute("PC", pc); | ||||||
|  |     trc->tr_handle.record_attribute("INSTR", instr_str); | ||||||
|  |     trc->tr_handle.record_attribute("MODE", lvl[cpu->get_mode()]); | ||||||
|  |     trc->tr_handle.record_attribute("MSTATUS", cpu->get_state()); | ||||||
|  |     trc->tr_handle.record_attribute("LTIME_START", quantum_keeper.get_current_time().value() / 1000); | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void core_complex::forward() { | ||||||
|  | #ifndef CWR_SYSTEMC | ||||||
|  | 	set_clock_period(clk_i.read()); | ||||||
|  | #else | ||||||
|  | 	set_clock_period(curr_clk.read()); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
|  |  | ||||||
| void core_complex::disass_output(uint64_t pc, const std::string instr_str) { | void core_complex::set_clock_period(sc_core::sc_time period) { | ||||||
| #ifdef WITH_SCV | 	curr_clk = period; | ||||||
|     if (m_db == nullptr) return; |     if (period == SC_ZERO_TIME) cpu->set_interrupt_execution(true); | ||||||
|     if (tr_handle.is_active()) tr_handle.end_transaction(); |  | ||||||
|     tr_handle = instr_tr_handle->begin_transaction(); |  | ||||||
|     tr_handle.record_attribute("PC", pc); |  | ||||||
|     tr_handle.record_attribute("INSTR", instr_str); |  | ||||||
|     tr_handle.record_attribute("MODE", lvl[cpu->get_mode()]); |  | ||||||
|     tr_handle.record_attribute("MSTATUS", cpu->get_state().mstatus.backing.val); |  | ||||||
|     tr_handle.record_attribute("LTIME_START", quantum_keeper.get_current_time().value() / 1000); |  | ||||||
| #endif |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void core_complex::clk_cb() { |  | ||||||
|     curr_clk = clk_i.read(); |  | ||||||
|     if (curr_clk == SC_ZERO_TIME) cpu->set_interrupt_execution(true); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void core_complex::rst_cb() { | void core_complex::rst_cb() { | ||||||
| @@ -360,14 +448,14 @@ void core_complex::run() { | |||||||
|     wait(SC_ZERO_TIME); // separate from elaboration phase |     wait(SC_ZERO_TIME); // separate from elaboration phase | ||||||
|     do { |     do { | ||||||
|         if (rst_i.read()) { |         if (rst_i.read()) { | ||||||
|             cpu->reset(reset_address.get_value()); |             cpu->reset(GET_PROP_VALUE(reset_address)); | ||||||
|             wait(rst_i.negedge_event()); |             wait(rst_i.negedge_event()); | ||||||
|         } |         } | ||||||
|         while (clk_i.read() == SC_ZERO_TIME) { |         while (curr_clk.read() == SC_ZERO_TIME) { | ||||||
|             wait(clk_i.value_changed_event()); |             wait(curr_clk.value_changed_event()); | ||||||
|         } |         } | ||||||
|         cpu->set_interrupt_execution(false); |         cpu->set_interrupt_execution(false); | ||||||
|         vm->start(); |         cpu->start(); | ||||||
|     } while (cpu->get_interrupt_execution()); |     } while (cpu->get_interrupt_execution()); | ||||||
|     sc_stop(); |     sc_stop(); | ||||||
| } | } | ||||||
| @@ -387,16 +475,14 @@ bool core_complex::read_mem(uint64_t addr, unsigned length, uint8_t *const data, | |||||||
|         gp.set_data_ptr(data); |         gp.set_data_ptr(data); | ||||||
|         gp.set_data_length(length); |         gp.set_data_length(length); | ||||||
|         gp.set_streaming_width(length); |         gp.set_streaming_width(length); | ||||||
|         sc_time delay{quantum_keeper.get_local_time()}; |         sc_time delay=quantum_keeper.get_local_time(); | ||||||
| #ifdef WITH_SCV |         if (trc->m_db != nullptr && trc->tr_handle.is_valid()) { | ||||||
|         if (m_db != nullptr && tr_handle.is_valid()) { |             if (is_fetch && trc->tr_handle.is_active()) { | ||||||
|             if (is_fetch && tr_handle.is_active()) { |                 trc->tr_handle.end_transaction(); | ||||||
|                 tr_handle.end_transaction(); |  | ||||||
|             } |             } | ||||||
|             auto preExt = new tlm::scc::scv::tlm_recording_extension(tr_handle, this); |             auto preExt = new tlm::scc::scv::tlm_recording_extension(trc->tr_handle, this); | ||||||
|             gp.set_extension(preExt); |             gp.set_extension(preExt); | ||||||
|         } |         } | ||||||
| #endif |  | ||||||
|         initiator->b_transport(gp, delay); |         initiator->b_transport(gp, delay); | ||||||
|         SCCTRACE(this->name()) << "read_mem(0x" << std::hex << addr << ") : " << data; |         SCCTRACE(this->name()) << "read_mem(0x" << std::hex << addr << ") : " << data; | ||||||
|         if (gp.get_response_status() != tlm::TLM_OK_RESPONSE) { |         if (gp.get_response_status() != tlm::TLM_OK_RESPONSE) { | ||||||
| @@ -436,13 +522,11 @@ bool core_complex::write_mem(uint64_t addr, unsigned length, const uint8_t *cons | |||||||
|         gp.set_data_ptr(write_buf.data()); |         gp.set_data_ptr(write_buf.data()); | ||||||
|         gp.set_data_length(length); |         gp.set_data_length(length); | ||||||
|         gp.set_streaming_width(length); |         gp.set_streaming_width(length); | ||||||
|         sc_time delay{quantum_keeper.get_local_time()}; |         sc_time delay=quantum_keeper.get_local_time(); | ||||||
| #ifdef WITH_SCV |         if (trc->m_db != nullptr && trc->tr_handle.is_valid()) { | ||||||
|         if (m_db != nullptr && tr_handle.is_valid()) { |             auto preExt = new tlm::scc::scv::tlm_recording_extension(trc->tr_handle, this); | ||||||
|             auto preExt = new tlm::scc::scv::tlm_recording_extension(tr_handle, this); |  | ||||||
|             gp.set_extension(preExt); |             gp.set_extension(preExt); | ||||||
|         } |         } | ||||||
| #endif |  | ||||||
|         initiator->b_transport(gp, delay); |         initiator->b_transport(gp, delay); | ||||||
|         quantum_keeper.set(delay); |         quantum_keeper.set(delay); | ||||||
|         SCCTRACE() << "write_mem(0x" << std::hex << addr << ") : " << data; |         SCCTRACE() << "write_mem(0x" << std::hex << addr << ") : " << data; | ||||||
| @@ -505,6 +589,5 @@ bool core_complex::write_mem_dbg(uint64_t addr, unsigned length, const uint8_t * | |||||||
|         return initiator->transport_dbg(gp) == length; |         return initiator->transport_dbg(gp) == length; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| } /* namespace SiFive */ | } /* namespace SiFive */ | ||||||
| } /* namespace sysc */ | } /* namespace sysc */ | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								src/vm/interp/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								src/vm/interp/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1 @@ | |||||||
| /vm_tgf_b.cpp | /vm_tgc_*.cpp | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Reference in New Issue
	
	Block a user