Compare commits
	
		
			2 Commits
		
	
	
		
			a53ee42e13
			...
			57347ae4d9
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 57347ae4d9 | |||
| 4876f18ba9 | 
| @@ -30,10 +30,10 @@ | ||||
|  * | ||||
|  *******************************************************************************/ | ||||
|  | ||||
| #include <iss/arch/${coreDef.name.toLowerCase()}.h> | ||||
| #include <iss/arch/riscv_hart_m_p.h> | ||||
| #include <iss/debugger/gdb_session.h> | ||||
| #include <iss/debugger/server.h> | ||||
| #include <iss/arch/${coreDef.name.toLowerCase()}.h> | ||||
| #include <iss/arch/riscv_hart_m_p.h> | ||||
| #include <iss/iss.h> | ||||
| #include <iss/llvm/vm_base.h> | ||||
| #include <util/logging.h> | ||||
|   | ||||
| @@ -477,10 +477,10 @@ template <typename BASE, features_e FEAT> std::pair<uint64_t, bool> riscv_hart_m | ||||
|     if (fp) { | ||||
|         std::array<char, 5> buf; | ||||
|         auto n = fread(buf.data(), 1, 4, fp); | ||||
|         fclose(fp); | ||||
|         if (n != 4) throw std::runtime_error("input file has insufficient size"); | ||||
|         buf[4] = 0; | ||||
|         if (strcmp(buf.data() + 1, "ELF") == 0) { | ||||
|             fclose(fp); | ||||
|             // Create elfio reader | ||||
|             ELFIO::elfio reader; | ||||
|             // Load ELF data | ||||
| @@ -571,12 +571,12 @@ iss::status riscv_hart_m_p<BASE, FEAT>::read(const address_type type, const acce | ||||
|             if (unlikely(is_fetch(access) && (addr&(alignment-1)))) { | ||||
|                 fault_data = addr; | ||||
|                 if (is_debug(access)) throw trap_access(0, addr); | ||||
|                 this->trap_state = (1 << 31); // issue trap 0 | ||||
|                 this->trap_state = (1UL << 31); // issue trap 0 | ||||
|                 return iss::Err; | ||||
|             } | ||||
|             try { | ||||
|                 if(!is_debug(access) && (addr&(alignment-1))){ | ||||
|                     this->trap_state = 1<<31 | 4<<16; | ||||
|                     this->trap_state = (1UL << 31) | 4<<16; | ||||
|                     fault_data=addr; | ||||
|                     return iss::Err; | ||||
|                 } | ||||
| @@ -595,12 +595,12 @@ iss::status riscv_hart_m_p<BASE, FEAT>::read(const address_type type, const acce | ||||
|                     res = read_mem( phys_addr, length, data); | ||||
|                 } | ||||
|                 if (unlikely(res != iss::Ok)){ | ||||
|                     this->trap_state = (1 << 31) | (5 << 16); // issue trap 5 (load access fault | ||||
|                     this->trap_state = (1UL << 31) | (5 << 16); // issue trap 5 (load access fault | ||||
|                     fault_data=addr; | ||||
|                 } | ||||
|                 return res; | ||||
|             } catch (trap_access &ta) { | ||||
|                 this->trap_state = (1 << 31) | ta.id; | ||||
|                 this->trap_state = (1UL << 31) | ta.id; | ||||
|                 fault_data=ta.addr; | ||||
|                 return iss::Err; | ||||
|             } | ||||
| @@ -626,7 +626,7 @@ iss::status riscv_hart_m_p<BASE, FEAT>::read(const address_type type, const acce | ||||
|         } | ||||
|         return iss::Ok; | ||||
|     } catch (trap_access &ta) { | ||||
|         this->trap_state = (1 << 31) | ta.id; | ||||
|         this->trap_state = (1UL << 31) | ta.id; | ||||
|         fault_data=ta.addr; | ||||
|         return iss::Err; | ||||
|     } | ||||
| @@ -664,12 +664,12 @@ iss::status riscv_hart_m_p<BASE, FEAT>::write(const address_type type, const acc | ||||
|             if (unlikely((access && iss::access_type::FETCH) && (addr & 0x1) == 1)) { | ||||
|                 fault_data = addr; | ||||
|                 if (access && iss::access_type::DEBUG) throw trap_access(0, addr); | ||||
|                 this->trap_state = (1 << 31); // issue trap 0 | ||||
|                 this->trap_state = (1UL << 31); // issue trap 0 | ||||
|                 return iss::Err; | ||||
|             } | ||||
|             try { | ||||
|                 if(length>1 && (addr&(length-1)) && (access&access_type::DEBUG) != access_type::DEBUG){ | ||||
|                     this->trap_state = 1<<31 | 6<<16; | ||||
|                     this->trap_state = (1UL << 31) | 6<<16; | ||||
|                     fault_data=addr; | ||||
|                     return iss::Err; | ||||
|                 } | ||||
| @@ -688,12 +688,12 @@ iss::status riscv_hart_m_p<BASE, FEAT>::write(const address_type type, const acc | ||||
|                     res = write_mem( phys_addr, length, data); | ||||
|                 } | ||||
|                 if (unlikely(res != iss::Ok)) { | ||||
|                     this->trap_state = (1 << 31) | (7 << 16); // issue trap 7 (Store/AMO access fault) | ||||
|                     this->trap_state = (1UL << 31) | (7 << 16); // issue trap 7 (Store/AMO access fault) | ||||
|                     fault_data=addr; | ||||
|                 } | ||||
|                 return res; | ||||
|             } catch (trap_access &ta) { | ||||
|                 this->trap_state = (1 << 31) | ta.id; | ||||
|                 this->trap_state = (1UL << 31) | ta.id; | ||||
|                 fault_data=ta.addr; | ||||
|                 return iss::Err; | ||||
|             } | ||||
| @@ -753,7 +753,7 @@ iss::status riscv_hart_m_p<BASE, FEAT>::write(const address_type type, const acc | ||||
|         } | ||||
|         return iss::Ok; | ||||
|     } catch (trap_access &ta) { | ||||
|         this->trap_state = (1 << 31) | ta.id; | ||||
|         this->trap_state = (1UL << 31) | ta.id; | ||||
|         fault_data=ta.addr; | ||||
|         return iss::Err; | ||||
|     } | ||||
|   | ||||
| @@ -526,10 +526,10 @@ template <typename BASE, features_e FEAT> std::pair<uint64_t, bool> riscv_hart_m | ||||
|     if (fp) { | ||||
|         std::array<char, 5> buf; | ||||
|         auto n = fread(buf.data(), 1, 4, fp); | ||||
|         fclose(fp); | ||||
|         if (n != 4) throw std::runtime_error("input file has insufficient size"); | ||||
|         buf[4] = 0; | ||||
|         if (strcmp(buf.data() + 1, "ELF") == 0) { | ||||
|             fclose(fp); | ||||
|             // Create elfio reader | ||||
|             ELFIO::elfio reader; | ||||
|             // Load ELF data | ||||
| @@ -708,7 +708,7 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::read(const address_type type, const acc | ||||
|                 if(!pmp_check(access, addr, length) && !is_debug(access)) { | ||||
|                     fault_data = addr; | ||||
|                     if (is_debug(access)) throw trap_access(0, addr); | ||||
|                     this->trap_state = (1 << 31) | ((access==access_type::FETCH?1:5) << 16); // issue trap 1 | ||||
|                     this->trap_state = (1UL << 31) | ((access==access_type::FETCH?1:5) << 16); // issue trap 1 | ||||
|                     return iss::Err; | ||||
|                 } | ||||
|             } | ||||
| @@ -716,12 +716,12 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::read(const address_type type, const acc | ||||
|             if (unlikely(is_fetch(access) && (addr&(alignment-1)))) { | ||||
|                 fault_data = addr; | ||||
|                 if (is_debug(access)) throw trap_access(0, addr); | ||||
|                 this->trap_state = (1 << 31); // issue trap 0 | ||||
|                 this->trap_state = (1UL << 31); // issue trap 0 | ||||
|                 return iss::Err; | ||||
|             } | ||||
|             try { | ||||
|                 if(!is_debug(access)  && (addr&(alignment-1))){ | ||||
|                     this->trap_state = 1<<31 | 4<<16; | ||||
|                     this->trap_state = (1UL << 31) | 4<<16; | ||||
|                     fault_data=addr; | ||||
|                     return iss::Err; | ||||
|                 } | ||||
| @@ -740,12 +740,12 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::read(const address_type type, const acc | ||||
|                     res = read_mem( phys_addr, length, data); | ||||
|                 } | ||||
|                 if (unlikely(res != iss::Ok)){ | ||||
|                     this->trap_state = (1 << 31) | (5 << 16); // issue trap 5 (load access fault | ||||
|                     this->trap_state = (1UL << 31) | (5 << 16); // issue trap 5 (load access fault | ||||
|                     fault_data=addr; | ||||
|                 } | ||||
|                 return res; | ||||
|             } catch (trap_access &ta) { | ||||
|                 this->trap_state = (1 << 31) | ta.id; | ||||
|                 this->trap_state = (1UL << 31) | ta.id; | ||||
|                 fault_data=ta.addr; | ||||
|                 return iss::Err; | ||||
|             } | ||||
| @@ -771,7 +771,7 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::read(const address_type type, const acc | ||||
|         } | ||||
|         return iss::Ok; | ||||
|     } catch (trap_access &ta) { | ||||
|         this->trap_state = (1 << 31) | ta.id; | ||||
|         this->trap_state = (1UL << 31) | ta.id; | ||||
|         fault_data=ta.addr; | ||||
|         return iss::Err; | ||||
|     } | ||||
| @@ -810,19 +810,19 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::write(const address_type type, const ac | ||||
|                 if(!pmp_check(access, addr, length) && (access&access_type::DEBUG) != access_type::DEBUG) { | ||||
|                     fault_data = addr; | ||||
|                     if (access && iss::access_type::DEBUG) throw trap_access(0, addr); | ||||
|                     this->trap_state = (1 << 31) | (7 << 16); // issue trap 1 | ||||
|                     this->trap_state = (1UL << 31) | (7 << 16); // issue trap 1 | ||||
|                     return iss::Err; | ||||
|                 } | ||||
|             } | ||||
|             if (unlikely((access && iss::access_type::FETCH) && (addr & 0x1) == 1)) { | ||||
|                 fault_data = addr; | ||||
|                 if (access && iss::access_type::DEBUG) throw trap_access(0, addr); | ||||
|                 this->trap_state = (1 << 31); // issue trap 0 | ||||
|                 this->trap_state = (1UL << 31); // issue trap 0 | ||||
|                 return iss::Err; | ||||
|             } | ||||
|             try { | ||||
|                 if(length>1 && (addr&(length-1)) && (access&access_type::DEBUG) != access_type::DEBUG){ | ||||
|                     this->trap_state = 1<<31 | 6<<16; | ||||
|                     this->trap_state = (1UL << 31) | 6<<16; | ||||
|                     fault_data=addr; | ||||
|                     return iss::Err; | ||||
|                 } | ||||
| @@ -841,12 +841,12 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::write(const address_type type, const ac | ||||
|                     res = write_mem( phys_addr, length, data); | ||||
|                 } | ||||
|                 if (unlikely(res != iss::Ok)) { | ||||
|                     this->trap_state = (1 << 31) | (7 << 16); // issue trap 7 (Store/AMO access fault) | ||||
|                     this->trap_state = (1UL << 31) | (7 << 16); // issue trap 7 (Store/AMO access fault) | ||||
|                     fault_data=addr; | ||||
|                 } | ||||
|                 return res; | ||||
|             } catch (trap_access &ta) { | ||||
|                 this->trap_state = (1 << 31) | ta.id; | ||||
|                 this->trap_state = (1UL << 31) | ta.id; | ||||
|                 fault_data=ta.addr; | ||||
|                 return iss::Err; | ||||
|             } | ||||
| @@ -906,7 +906,7 @@ iss::status riscv_hart_mu_p<BASE, FEAT>::write(const address_type type, const ac | ||||
|         } | ||||
|         return iss::Ok; | ||||
|     } catch (trap_access &ta) { | ||||
|         this->trap_state = (1 << 31) | ta.id; | ||||
|         this->trap_state = (1UL << 31) | ta.id; | ||||
|         fault_data=ta.addr; | ||||
|         return iss::Err; | ||||
|     } | ||||
|   | ||||
| @@ -45,7 +45,7 @@ namespace iss { | ||||
|  | ||||
| namespace plugin { | ||||
|  | ||||
| class cycle_estimate: public iss::vm_plugin { | ||||
| class cycle_estimate: public vm_plugin { | ||||
| 	BEGIN_BF_DECL(instr_desc, uint32_t) | ||||
| 		BF_FIELD(taken, 24, 8) | ||||
| 		BF_FIELD(not_taken, 16, 8) | ||||
|   | ||||
							
								
								
									
										471
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										471
									
								
								src/main.cpp
									
									
									
									
									
								
							| @@ -1,236 +1,241 @@ | ||||
| /******************************************************************************* | ||||
|  * Copyright (C) 2017, 2018 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. | ||||
|  * | ||||
|  *******************************************************************************/ | ||||
|  | ||||
| #include <iostream> | ||||
| #include <iss/factory.h> | ||||
|  | ||||
| #include <boost/lexical_cast.hpp> | ||||
| #include <boost/program_options.hpp> | ||||
| #include "iss/arch/tgc_mapper.h" | ||||
| #ifdef WITH_LLVM | ||||
| #include <iss/llvm/jit_helper.h> | ||||
| #endif | ||||
| #include <iss/log_categories.h> | ||||
| #include "iss/plugin/cycle_estimate.h" | ||||
| #include "iss/plugin/instruction_count.h" | ||||
| #include "iss/plugin/pctrace.h" | ||||
| #include <iss/plugin/loader.h> | ||||
| #if defined(HAS_LUA) | ||||
| #include <iss/plugin/lua.h> | ||||
| #endif | ||||
|  | ||||
| namespace po = boost::program_options; | ||||
|  | ||||
| int main(int argc, char *argv[]) { | ||||
|     /* | ||||
|      *  Define and parse the program options | ||||
|      */ | ||||
|     po::variables_map clim; | ||||
|     po::options_description desc("Options"); | ||||
|     // clang-format off | ||||
|     desc.add_options() | ||||
|         ("help,h", "Print help message") | ||||
|         ("verbose,v", po::value<int>()->implicit_value(0), "Sets logging verbosity") | ||||
|         ("logfile,l", po::value<std::string>(), "Sets default log file.") | ||||
|         ("disass,d", po::value<std::string>()->implicit_value(""), "Enables disassembly") | ||||
|         ("gdb-port,g", po::value<unsigned>()->default_value(0), "enable gdb server and specify port to use") | ||||
|         ("instructions,i", po::value<uint64_t>()->default_value(std::numeric_limits<uint64_t>::max()), "max. number of instructions to simulate") | ||||
|         ("reset,r", po::value<std::string>(), "reset address") | ||||
|         ("dump-ir", "dump the intermediate representation") | ||||
|         ("elf,f", po::value<std::vector<std::string>>(), "ELF file(s) to load") | ||||
|         ("mem,m", po::value<std::string>(), "the memory input file") | ||||
|         ("plugin,p", po::value<std::vector<std::string>>(), "plugin to activate") | ||||
|         ("backend", po::value<std::string>()->default_value("interp"), "the memory input file") | ||||
|         ("isa", po::value<std::string>()->default_value("tgc_c"), "isa to use for simulation"); | ||||
|     // clang-format on | ||||
|     auto parsed = po::command_line_parser(argc, argv).options(desc).allow_unregistered().run(); | ||||
|     try { | ||||
|         po::store(parsed, clim); // can throw | ||||
|         // --help option | ||||
|         if (clim.count("help")) { | ||||
|             std::cout << "DBT-RISE-RiscV simulator for RISC-V" << std::endl << desc << std::endl; | ||||
|             return 0; | ||||
|         } | ||||
|         po::notify(clim); // throws on error, so do after help in case | ||||
|     } catch (po::error &e) { | ||||
|         // there are problems | ||||
|         std::cerr << "ERROR: " << e.what() << std::endl << std::endl; | ||||
|         std::cerr << desc << std::endl; | ||||
|         return 1; | ||||
|     } | ||||
|     std::vector<std::string> args = collect_unrecognized(parsed.options, po::include_positional); | ||||
|  | ||||
|     LOGGER(DEFAULT)::print_time() = false; | ||||
|     LOGGER(connection)::print_time() = false; | ||||
|     if (clim.count("verbose")) { | ||||
|         auto l = logging::as_log_level(clim["verbose"].as<int>()); | ||||
|         LOGGER(DEFAULT)::reporting_level() = l; | ||||
|         LOGGER(connection)::reporting_level() = l; | ||||
|     } | ||||
|     if (clim.count("logfile")) { | ||||
|         // configure the connection logger | ||||
|         auto f = fopen(clim["logfile"].as<std::string>().c_str(), "w"); | ||||
|         LOG_OUTPUT(DEFAULT)::stream() = f; | ||||
|         LOG_OUTPUT(connection)::stream() = f; | ||||
|     } | ||||
|  | ||||
|     std::vector<iss::vm_plugin *> plugin_list; | ||||
|     auto res = 0; | ||||
|     try { | ||||
| #ifdef WITH_LLVM | ||||
|         // application code comes here // | ||||
|         iss::init_jit_debug(argc, argv); | ||||
| #endif | ||||
|         bool dump = clim.count("dump-ir"); | ||||
|         // instantiate the simulator | ||||
|         iss::vm_ptr vm{nullptr}; | ||||
|         iss::cpu_ptr cpu{nullptr}; | ||||
|         std::string isa_opt(clim["isa"].as<std::string>()); | ||||
|         if (isa_opt == "tgc_c") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 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 | ||||
| #endif | ||||
| #ifdef CORE_TGC_C_XRB_NN | ||||
|         if (isa_opt == "tgc_c_xrb_nn") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_c_xrb_nn_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_D | ||||
|         if (isa_opt == "tgc_d") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_d_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_D_XRB_MAC | ||||
|         if (isa_opt == "tgc_d_xrb_mac") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_d_xrb_mac_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_D_XRB_NN | ||||
|         if (isa_opt == "tgc_d_xrb_nn") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_d_xrb_nn_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_E | ||||
|         if (isa_opt == "tgc_e") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_e_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
|         { | ||||
| /******************************************************************************* | ||||
|  * Copyright (C) 2017, 2018 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. | ||||
|  * | ||||
|  *******************************************************************************/ | ||||
|  | ||||
| #include <iostream> | ||||
| #include "iss/factory.h" | ||||
|  | ||||
| #include <boost/lexical_cast.hpp> | ||||
| #include <boost/program_options.hpp> | ||||
| #include "iss/arch/tgc_mapper.h" | ||||
| #ifdef WITH_LLVM | ||||
| #include <iss/llvm/jit_helper.h> | ||||
| #endif | ||||
| #include <iss/log_categories.h> | ||||
| #include "iss/plugin/cycle_estimate.h" | ||||
| #include "iss/plugin/instruction_count.h" | ||||
| #include "iss/plugin/pctrace.h" | ||||
| #ifndef WIN32 | ||||
| #include <iss/plugin/loader.h> | ||||
| #endif | ||||
| #if defined(HAS_LUA) | ||||
| #include <iss/plugin/lua.h> | ||||
| #endif | ||||
|  | ||||
| namespace po = boost::program_options; | ||||
|  | ||||
| int main(int argc, char *argv[]) { | ||||
|     /* | ||||
|      *  Define and parse the program options | ||||
|      */ | ||||
|     po::variables_map clim; | ||||
|     po::options_description desc("Options"); | ||||
|     // clang-format off | ||||
|     desc.add_options() | ||||
|         ("help,h", "Print help message") | ||||
|         ("verbose,v", po::value<int>()->implicit_value(0), "Sets logging verbosity") | ||||
|         ("logfile,l", po::value<std::string>(), "Sets default log file.") | ||||
|         ("disass,d", po::value<std::string>()->implicit_value(""), "Enables disassembly") | ||||
|         ("gdb-port,g", po::value<unsigned>()->default_value(0), "enable gdb server and specify port to use") | ||||
|         ("instructions,i", po::value<uint64_t>()->default_value(std::numeric_limits<uint64_t>::max()), "max. number of instructions to simulate") | ||||
|         ("reset,r", po::value<std::string>(), "reset address") | ||||
|         ("dump-ir", "dump the intermediate representation") | ||||
|         ("elf,f", po::value<std::vector<std::string>>(), "ELF file(s) to load") | ||||
|         ("mem,m", po::value<std::string>(), "the memory input file") | ||||
|         ("plugin,p", po::value<std::vector<std::string>>(), "plugin to activate") | ||||
|         ("backend", po::value<std::string>()->default_value("interp"), "the memory input file") | ||||
|         ("isa", po::value<std::string>()->default_value("tgc_c"), "isa to use for simulation"); | ||||
|     // clang-format on | ||||
|     auto parsed = po::command_line_parser(argc, argv).options(desc).allow_unregistered().run(); | ||||
|     try { | ||||
|         po::store(parsed, clim); // can throw | ||||
|         // --help option | ||||
|         if (clim.count("help")) { | ||||
|             std::cout << "DBT-RISE-RiscV simulator for RISC-V" << std::endl << desc << std::endl; | ||||
|             return 0; | ||||
|         } | ||||
|         po::notify(clim); // throws on error, so do after help in case | ||||
|     } catch (po::error &e) { | ||||
|         // there are problems | ||||
|         std::cerr << "ERROR: " << e.what() << std::endl << std::endl; | ||||
|         std::cerr << desc << std::endl; | ||||
|         return 1; | ||||
|     } | ||||
|     std::vector<std::string> args = collect_unrecognized(parsed.options, po::include_positional); | ||||
|  | ||||
|     LOGGER(DEFAULT)::print_time() = false; | ||||
|     LOGGER(connection)::print_time() = false; | ||||
|     if (clim.count("verbose")) { | ||||
|         auto l = logging::as_log_level(clim["verbose"].as<int>()); | ||||
|         LOGGER(DEFAULT)::reporting_level() = l; | ||||
|         LOGGER(connection)::reporting_level() = l; | ||||
|     } | ||||
|     if (clim.count("logfile")) { | ||||
|         // configure the connection logger | ||||
|         auto f = fopen(clim["logfile"].as<std::string>().c_str(), "w"); | ||||
|         LOG_OUTPUT(DEFAULT)::stream() = f; | ||||
|         LOG_OUTPUT(connection)::stream() = f; | ||||
|     } | ||||
|  | ||||
|     std::vector<iss::vm_plugin *> plugin_list; | ||||
|     auto res = 0; | ||||
|     try { | ||||
| #ifdef WITH_LLVM | ||||
|         // application code comes here // | ||||
|         iss::init_jit_debug(argc, argv); | ||||
| #endif | ||||
|         bool dump = clim.count("dump-ir"); | ||||
|         // instantiate the simulator | ||||
|         iss::vm_ptr vm{nullptr}; | ||||
|         iss::cpu_ptr cpu{nullptr}; | ||||
|         std::string isa_opt(clim["isa"].as<std::string>()); | ||||
|         if (isa_opt == "tgc_c") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 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 | ||||
| #endif | ||||
| #ifdef CORE_TGC_C_XRB_NN | ||||
|         if (isa_opt == "tgc_c_xrb_nn") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_c_xrb_nn_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_D | ||||
|         if (isa_opt == "tgc_d") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_d_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_D_XRB_MAC | ||||
|         if (isa_opt == "tgc_d_xrb_mac") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_d_xrb_mac_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_D_XRB_NN | ||||
|         if (isa_opt == "tgc_d_xrb_nn") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_d_xrb_nn_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
| #ifdef CORE_TGC_E | ||||
|         if (isa_opt == "tgc_e") { | ||||
|             std::tie(cpu, vm) = | ||||
|                 iss::create_cpu<tgc_e_plat_type>(clim["backend"].as<std::string>(), clim["gdb-port"].as<unsigned>()); | ||||
|         } else | ||||
| #endif | ||||
|         { | ||||
|             LOG(ERR) << "Illegal argument value for '--isa': " << isa_opt << std::endl; | ||||
|             return 127; | ||||
|         } | ||||
|         if (clim.count("plugin")) { | ||||
|             for (std::string const& opt_val : clim["plugin"].as<std::vector<std::string>>()) { | ||||
|                 std::string plugin_name=opt_val; | ||||
|                 std::string filename{"cycles.txt"}; | ||||
|                 std::size_t found = opt_val.find('='); | ||||
|                 if (found != std::string::npos) { | ||||
|                     plugin_name = opt_val.substr(0, found); | ||||
|                     filename = opt_val.substr(found + 1, opt_val.size()); | ||||
|                 } | ||||
|                 if (plugin_name == "ic") { | ||||
|                     auto *ic_plugin = new iss::plugin::instruction_count(filename); | ||||
|                     vm->register_plugin(*ic_plugin); | ||||
|                     plugin_list.push_back(ic_plugin); | ||||
|                 } else if (plugin_name == "ce") { | ||||
|                     auto *ce_plugin = new iss::plugin::cycle_estimate(filename); | ||||
|                     vm->register_plugin(*ce_plugin); | ||||
|                     plugin_list.push_back(ce_plugin); | ||||
|                 } else if (plugin_name == "pctrace") { | ||||
|                     auto *plugin = new iss::plugin::pctrace(filename); | ||||
|                     vm->register_plugin(*plugin); | ||||
|                     plugin_list.push_back(plugin); | ||||
|                } else { | ||||
|                     std::array<char const*, 1> a{{filename.c_str()}}; | ||||
|                     iss::plugin::loader l(plugin_name, {{"initPlugin"}}); | ||||
|                     auto* plugin = l.call_function<iss::vm_plugin*>("initPlugin", a.size(), a.data()); | ||||
|                     if(plugin){ | ||||
|                         vm->register_plugin(*plugin); | ||||
|                         plugin_list.push_back(plugin); | ||||
|                     } else { | ||||
|             return 127; | ||||
|         } | ||||
|         if (clim.count("plugin")) { | ||||
|             for (std::string const& opt_val : clim["plugin"].as<std::vector<std::string>>()) { | ||||
|                 std::string plugin_name=opt_val; | ||||
|                 std::string filename{"cycles.txt"}; | ||||
|                 std::size_t found = opt_val.find('='); | ||||
|                 if (found != std::string::npos) { | ||||
|                     plugin_name = opt_val.substr(0, found); | ||||
|                     filename = opt_val.substr(found + 1, opt_val.size()); | ||||
|                 } | ||||
|                 if (plugin_name == "ic") { | ||||
|                     auto *ic_plugin = new iss::plugin::instruction_count(filename); | ||||
|                     vm->register_plugin(*ic_plugin); | ||||
|                     plugin_list.push_back(ic_plugin); | ||||
|                 } else if (plugin_name == "ce") { | ||||
|                     auto *ce_plugin = new iss::plugin::cycle_estimate(filename); | ||||
|                     vm->register_plugin(*ce_plugin); | ||||
|                     plugin_list.push_back(ce_plugin); | ||||
|                 } else if (plugin_name == "pctrace") { | ||||
|                     auto *plugin = new iss::plugin::pctrace(filename); | ||||
|                     vm->register_plugin(*plugin); | ||||
|                     plugin_list.push_back(plugin); | ||||
|                } else { | ||||
| #ifndef WIN32 | ||||
|                     std::array<char const*, 1> a{{filename.c_str()}}; | ||||
|                     iss::plugin::loader l(plugin_name, {{"initPlugin"}}); | ||||
|                     auto* plugin = l.call_function<iss::vm_plugin*>("initPlugin", a.size(), a.data()); | ||||
|                     if(plugin){ | ||||
|                         vm->register_plugin(*plugin); | ||||
|                         plugin_list.push_back(plugin); | ||||
|                     } else | ||||
| #endif | ||||
|                     { | ||||
|                         LOG(ERR) << "Unknown plugin name: " << plugin_name << ", valid names are 'ce', 'ic'" << std::endl; | ||||
|                         return 127; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (clim.count("disass")) { | ||||
|             vm->setDisassEnabled(true); | ||||
|             LOGGER(disass)::reporting_level() = logging::INFO; | ||||
|             LOGGER(disass)::print_time() = false; | ||||
|             auto file_name = clim["disass"].as<std::string>(); | ||||
|             if (file_name.length() > 0) { | ||||
|                 LOG_OUTPUT(disass)::stream() = fopen(file_name.c_str(), "w"); | ||||
|                 LOGGER(disass)::print_severity() = false; | ||||
|             } | ||||
|         } | ||||
|         uint64_t start_address = 0; | ||||
|         if (clim.count("mem")) | ||||
|             vm->get_arch()->load_file(clim["mem"].as<std::string>()); | ||||
|         if (clim.count("elf")) | ||||
|             for (std::string input : clim["elf"].as<std::vector<std::string>>()) { | ||||
|                 auto start_addr = vm->get_arch()->load_file(input); | ||||
|                 if (start_addr.second) start_address = start_addr.first; | ||||
|             } | ||||
|         for (std::string input : args) { | ||||
|             auto start_addr = vm->get_arch()->load_file(input); // treat remaining arguments as elf files | ||||
|             if (start_addr.second) start_address = start_addr.first; | ||||
|         } | ||||
|         if (clim.count("reset")) { | ||||
|             auto str = clim["reset"].as<std::string>(); | ||||
|             start_address = str.find("0x") == 0 ? std::stoull(str.substr(2), nullptr, 16) : std::stoull(str, nullptr, 10); | ||||
|         } | ||||
|         vm->reset(start_address); | ||||
|         auto cycles = clim["instructions"].as<uint64_t>(); | ||||
|         res = vm->start(cycles, dump); | ||||
|     } catch (std::exception &e) { | ||||
|                         return 127; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         if (clim.count("disass")) { | ||||
|             vm->setDisassEnabled(true); | ||||
|             LOGGER(disass)::reporting_level() = logging::INFO; | ||||
|             LOGGER(disass)::print_time() = false; | ||||
|             auto file_name = clim["disass"].as<std::string>(); | ||||
|             if (file_name.length() > 0) { | ||||
|                 LOG_OUTPUT(disass)::stream() = fopen(file_name.c_str(), "w"); | ||||
|                 LOGGER(disass)::print_severity() = false; | ||||
|             } | ||||
|         } | ||||
|         uint64_t start_address = 0; | ||||
|         if (clim.count("mem")) | ||||
|             vm->get_arch()->load_file(clim["mem"].as<std::string>()); | ||||
|         if (clim.count("elf")) | ||||
|             for (std::string input : clim["elf"].as<std::vector<std::string>>()) { | ||||
|                 auto start_addr = vm->get_arch()->load_file(input); | ||||
|                 if (start_addr.second) start_address = start_addr.first; | ||||
|             } | ||||
|         for (std::string input : args) { | ||||
|             auto start_addr = vm->get_arch()->load_file(input); // treat remaining arguments as elf files | ||||
|             if (start_addr.second) start_address = start_addr.first; | ||||
|         } | ||||
|         if (clim.count("reset")) { | ||||
|             auto str = clim["reset"].as<std::string>(); | ||||
|             start_address = str.find("0x") == 0 ? std::stoull(str.substr(2), nullptr, 16) : std::stoull(str, nullptr, 10); | ||||
|         } | ||||
|         vm->reset(start_address); | ||||
|         auto cycles = clim["instructions"].as<uint64_t>(); | ||||
|         res = vm->start(cycles, dump); | ||||
|     } catch (std::exception &e) { | ||||
|         LOG(ERR) << "Unhandled Exception reached the top of main: " << e.what() << ", application will now exit" | ||||
|                    << std::endl; | ||||
|         res = 2; | ||||
|     } | ||||
|     // cleanup to let plugins report of needed | ||||
|     for (auto *p : plugin_list) { | ||||
|         delete p; | ||||
|     } | ||||
|     return res; | ||||
| } | ||||
|                    << std::endl; | ||||
|         res = 2; | ||||
|     } | ||||
|     // cleanup to let plugins report of needed | ||||
|     for (auto *p : plugin_list) { | ||||
|         delete p; | ||||
|     } | ||||
|     return res; | ||||
| } | ||||
|   | ||||
| @@ -30,14 +30,16 @@ | ||||
|  * | ||||
|  *******************************************************************************/ | ||||
|  | ||||
| // clang-format off | ||||
| #include <iss/debugger/gdb_session.h> | ||||
| #include <iss/debugger/encoderdecoder.h> | ||||
| #include <iss/debugger/server.h> | ||||
| #include <iss/debugger/target_adapter_if.h> | ||||
| #include <iss/iss.h> | ||||
| // clang-format off | ||||
| #include <iss/debugger/gdb_session.h> | ||||
| #include <iss/debugger/encoderdecoder.h> | ||||
| #include <iss/debugger/server.h> | ||||
| #include <iss/debugger/target_adapter_if.h> | ||||
| #include <iss/iss.h> | ||||
| #include <iss/vm_types.h> | ||||
| #include <iss/plugin/loader.h> | ||||
| #ifndef WIN32 | ||||
| #include <iss/plugin/loader.h> | ||||
| #endif | ||||
| #include "core_complex.h" | ||||
| #include <iss/arch/tgc_mapper.h> | ||||
| #include <scc/report.h> | ||||
| @@ -49,7 +51,7 @@ | ||||
| #include <iss/plugin/instruction_count.h> | ||||
| #include <iss/plugin/pctrace.h> | ||||
|  | ||||
| // clang-format on | ||||
| // clang-format on | ||||
|  | ||||
| #define STR(X) #X | ||||
| #define CREATE_CORE(CN) \ | ||||
| @@ -68,12 +70,12 @@ using namespace scv_tr; | ||||
| #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 | ||||
|  | ||||
| #ifdef _MSC_VER | ||||
| // not #if defined(_WIN32) || defined(_WIN64) because we have strncasecmp in mingw | ||||
| #define strncasecmp _strnicmp | ||||
| #define strcasecmp _stricmp | ||||
| #endif | ||||
|  | ||||
| namespace sysc { | ||||
| namespace tgfs { | ||||
| using namespace std; | ||||
| @@ -296,7 +298,7 @@ public: | ||||
|         CREATE_CORE(tgc_d_xrb_nn) | ||||
| #endif | ||||
|         { | ||||
|             LOG(ERR) << "Illegal argument value for core type: " << type << std::endl; | ||||
|             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(); | ||||
| @@ -409,6 +411,7 @@ void core_complex::before_end_of_elaboration() { | ||||
|                 cpu->vm->register_plugin(*plugin); | ||||
|                 plugin_list.push_back(plugin); | ||||
|             } else { | ||||
| #ifndef WIN32 | ||||
|                 std::array<char const*, 1> a{{filename.c_str()}}; | ||||
|                 iss::plugin::loader l(plugin_name, {{"initPlugin"}}); | ||||
|                 auto* plugin = l.call_function<iss::vm_plugin*>("initPlugin", a.size(), a.data()); | ||||
| @@ -416,6 +419,7 @@ void core_complex::before_end_of_elaboration() { | ||||
|                     cpu->vm->register_plugin(*plugin); | ||||
|                     plugin_list.push_back(plugin); | ||||
|                 } else | ||||
| #endif | ||||
|                     SCCERR(SCMOD) << "Unknown plugin '" << plugin_name << "' or plugin not found"; | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -30,11 +30,10 @@ | ||||
|  * | ||||
|  *******************************************************************************/ | ||||
|  | ||||
| #include <vm/fp_functions.h> | ||||
| #include <iss/arch/tgc_c.h> | ||||
| #include <iss/arch/riscv_hart_m_p.h> | ||||
| #include <iss/debugger/gdb_session.h> | ||||
| #include <iss/debugger/server.h> | ||||
| #include <iss/arch/tgc_c.h> | ||||
| #include <iss/arch/riscv_hart_m_p.h> | ||||
| #include <iss/iss.h> | ||||
| #include <iss/interp/vm_base.h> | ||||
| #include <util/logging.h> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user