Compare commits
	
		
			2 Commits
		
	
	
		
			52ed8b81a6
			...
			1c90fe765d
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1c90fe765d | |||
| 1720bd4aaa | 
							
								
								
									
										180
									
								
								src/plugin/pctrace.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										180
									
								
								src/plugin/pctrace.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,180 @@ | |||||||
|  | #include <iss/arch_if.h> | ||||||
|  | #include <iss/plugin/pctrace.h> | ||||||
|  | #include <util/logging.h> | ||||||
|  | #include <util/ities.h> | ||||||
|  | #include <rapidjson/document.h> | ||||||
|  | #include <rapidjson/istreamwrapper.h> | ||||||
|  | #include "rapidjson/writer.h" | ||||||
|  | #include "rapidjson/stringbuffer.h" | ||||||
|  | #include <rapidjson/ostreamwrapper.h> | ||||||
|  | #include <rapidjson/error/en.h> | ||||||
|  | #include <fstream> | ||||||
|  | #include <iostream> | ||||||
|  | #ifdef WITH_LZ4 | ||||||
|  | #include <lz4frame.h> | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | namespace iss { | ||||||
|  | namespace plugin { | ||||||
|  |  | ||||||
|  | using namespace rapidjson; | ||||||
|  | using namespace std; | ||||||
|  |  | ||||||
|  | #ifdef WITH_LZ4 | ||||||
|  | class lz4compress_steambuf: public std::streambuf { | ||||||
|  | public: | ||||||
|  |     lz4compress_steambuf(const lz4compress_steambuf&) = delete; | ||||||
|  |     lz4compress_steambuf& operator=(const lz4compress_steambuf&) = delete; | ||||||
|  |     lz4compress_steambuf(std::ostream &sink, size_t buf_size) | ||||||
|  |     : sink(sink) | ||||||
|  |     , src_buf(buf_size) | ||||||
|  |     , dest_buf(LZ4F_compressBound(buf_size, nullptr)) | ||||||
|  |     { | ||||||
|  |         auto errCode = LZ4F_createCompressionContext(&ctx, LZ4F_VERSION); | ||||||
|  |         if (LZ4F_isError(errCode) != 0) | ||||||
|  |             throw std::runtime_error(std::string("Failed to create LZ4 context: ") + LZ4F_getErrorName(errCode)); | ||||||
|  |         size_t ret = LZ4F_compressBegin(ctx, &dest_buf.front(), dest_buf.capacity(), nullptr); | ||||||
|  |         if (LZ4F_isError(ret) != 0) | ||||||
|  |             throw std::runtime_error(std::string("Failed to start LZ4 compression: ") + LZ4F_getErrorName(ret)); | ||||||
|  |         setp(src_buf.data(), src_buf.data() + src_buf.size() - 1); | ||||||
|  |         sink.write(dest_buf.data(), ret); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     ~lz4compress_steambuf() { | ||||||
|  |         close(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void close() { | ||||||
|  |         if (closed) | ||||||
|  |             return; | ||||||
|  |         sync(); | ||||||
|  |         auto ret = LZ4F_compressEnd(ctx, dest_buf.data(), dest_buf.capacity(), nullptr); | ||||||
|  |         if (LZ4F_isError(ret) != 0) | ||||||
|  |             throw std::runtime_error(std::string("Failed to finish LZ4 compression: ") + LZ4F_getErrorName(ret)); | ||||||
|  |         sink.write(dest_buf.data(), ret); | ||||||
|  |         LZ4F_freeCompressionContext(ctx); | ||||||
|  |         closed = true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     int_type overflow(int_type ch) override { | ||||||
|  |         compress_and_write(); | ||||||
|  |         *pptr() = static_cast<char_type>(ch); | ||||||
|  |         pbump(1); | ||||||
|  |         return ch; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     int_type sync() override { | ||||||
|  |         compress_and_write(); | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     void compress_and_write() { | ||||||
|  |         if (closed) | ||||||
|  |             throw std::runtime_error("Cannot write to closed stream"); | ||||||
|  |         if(auto orig_size = pptr() - pbase()){ | ||||||
|  |             auto ret = LZ4F_compressUpdate(ctx, dest_buf.data(), dest_buf.capacity(), pbase(), orig_size, nullptr); | ||||||
|  |             if (LZ4F_isError(ret) != 0) | ||||||
|  |                 throw std::runtime_error(std::string("LZ4 compression failed: ") + LZ4F_getErrorName(ret)); | ||||||
|  |             if(ret) sink.write(dest_buf.data(), ret); | ||||||
|  |             pbump(-orig_size); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     std::ostream &sink; | ||||||
|  |     std::vector<char> src_buf; | ||||||
|  |     std::vector<char> dest_buf; | ||||||
|  |     LZ4F_compressionContext_t ctx{ nullptr }; | ||||||
|  |     bool closed{ false }; | ||||||
|  | }; | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | cov::cov(std::string const &filename) | ||||||
|  | : instr_if(nullptr) | ||||||
|  | , filename(filename) | ||||||
|  | , output("output.trc") | ||||||
|  | #ifdef WITH_LZ4 | ||||||
|  | , strbuf(new lz4compress_steambuf(output, 4096)) | ||||||
|  | , ostr(strbuf.get()) | ||||||
|  | #endif | ||||||
|  | { } | ||||||
|  |  | ||||||
|  | cov::~cov() { } | ||||||
|  |  | ||||||
|  | bool cov::registration(const char *const version, vm_if& vm) { | ||||||
|  |     instr_if = vm.get_arch()->get_instrumentation_if(); | ||||||
|  |     if(!instr_if) return false; | ||||||
|  |     const string  core_name = instr_if->core_type_name(); | ||||||
|  |     if (filename.length() > 0) { | ||||||
|  |         ifstream is(filename); | ||||||
|  |         if (is.is_open()) { | ||||||
|  |             try { | ||||||
|  |                 IStreamWrapper isw(is); | ||||||
|  |                 Document d; | ||||||
|  |                 ParseResult ok = d.ParseStream(isw); | ||||||
|  |                 if(ok) { | ||||||
|  |                     Value& val = d[core_name.c_str()]; | ||||||
|  |                     if(val.IsArray()){ | ||||||
|  |                         delays.reserve(val.Size()); | ||||||
|  |                         for (auto it = val.Begin(); it != val.End(); ++it) { | ||||||
|  |                             auto& name = (*it)["name"]; | ||||||
|  |                             auto& size = (*it)["size"]; | ||||||
|  |                             auto& delay = (*it)["delay"]; | ||||||
|  |                             auto& branch = (*it)["branch"]; | ||||||
|  |                             if(delay.IsArray()) { | ||||||
|  |                                 auto dt = delay[0].Get<unsigned>(); | ||||||
|  |                                 auto dnt = delay[1].Get<unsigned>(); | ||||||
|  |                                 delays.push_back(instr_desc{size.Get<unsigned>(), dt, dnt, branch.Get<bool>()}); | ||||||
|  |                             } else if(delay.Is<unsigned>()) { | ||||||
|  |                                 auto d = delay.Get<unsigned>(); | ||||||
|  |                                 delays.push_back(instr_desc{size.Get<unsigned>(), d, d, branch.Get<bool>()}); | ||||||
|  |                             } else | ||||||
|  |                                 throw runtime_error("JSON parse error"); | ||||||
|  |  | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                         LOG(ERR)<<"plugin cycle_estimate: could not find an entry for "<<core_name<<" in JSON file"<<endl; | ||||||
|  |                         return false; | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     LOG(ERR)<<"plugin cycle_estimate: could not parse in JSON file at "<< ok.Offset()<<": "<<GetParseError_En(ok.Code())<<endl; | ||||||
|  |                     return false; | ||||||
|  |                 } | ||||||
|  |             } catch (runtime_error &e) { | ||||||
|  |                 LOG(ERR) << "Could not parse input file " << filename << ", reason: " << e.what(); | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             LOG(ERR) << "Could not open input file " << filename; | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void cov::callback(instr_info_t iinfo, const exec_info& einfo) { | ||||||
|  |     auto delay = 0; | ||||||
|  |     size_t id = iinfo.instr_id; | ||||||
|  |     auto entry = delays[id]; | ||||||
|  |     auto instr = instr_if->get_instr_word(); | ||||||
|  |     auto call = id==65 || id ==86 || ((id==2 || id==3) && bit_sub<7,5>(instr)!=0) ;//not taking care of tail calls (jalr with loading x6) | ||||||
|  |     bool taken = einfo.branch_taken; | ||||||
|  |     bool compressed = (instr&0x3)!=0x3; | ||||||
|  |     if (einfo.branch_taken) { | ||||||
|  |         delay = entry.taken; | ||||||
|  |         if(entry.taken > 1) | ||||||
|  |             instr_if->set_curr_instr_cycles(entry.taken); | ||||||
|  |     } else { | ||||||
|  |         delay = entry.not_taken; | ||||||
|  |         if (entry.not_taken > 1) | ||||||
|  |             instr_if->set_curr_instr_cycles(entry.not_taken); | ||||||
|  |     } | ||||||
|  | #ifndef WITH_LZ4 | ||||||
|  |     output<<std::hex <<"0x" << instr_if->get_pc() <<"," << delay <<"," << call<<","<<(compressed?2:4) <<"\n"; | ||||||
|  | #else | ||||||
|  |     auto rdbuf=ostr.rdbuf(); | ||||||
|  |     ostr<<std::hex <<"0x" << instr_if->get_pc() <<"," << delay <<"," << call<<","<<(compressed?2:4) <<"\n"; | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  | } | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user