mirror of
				https://github.com/TomHarte/CLK.git
				synced 2025-10-26 01:23:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			198 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //
 | |
| //  CachingExecutor.hpp
 | |
| //  Clock Signal
 | |
| //
 | |
| //  Created by Thomas Harte on 16/01/2021.
 | |
| //  Copyright © 2021 Thomas Harte. All rights reserved.
 | |
| //
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include "Numeric/Sizes.hpp"
 | |
| 
 | |
| #include <array>
 | |
| #include <cstdint>
 | |
| #include <limits>
 | |
| #include <list>
 | |
| #include <map>
 | |
| #include <queue>
 | |
| #include <unordered_map>
 | |
| 
 | |
| namespace InstructionSet {
 | |
| 
 | |
| /*!
 | |
| 	A caching executor makes use of an instruction set-specific executor to cache 'performers' (i.e. function pointers)
 | |
| 	that result from decoding.
 | |
| 
 | |
| 	In other words, it's almost a JIT compiler, but producing threaded code (in the Forth sense) and then incurring whatever
 | |
| 	costs sit behind using the C ABI for calling. Since there'll always be exactly one parameter, being the specific executor,
 | |
| 	hopefully the calling costs are acceptable.
 | |
| 
 | |
| 	Intended usage is for specific executors to subclass from this and declare it a friend.
 | |
| 
 | |
| 	TODO: determine promises re: interruption, amongst other things.
 | |
| */
 | |
| template <
 | |
| 	/// Indicates the Executor for this platform.
 | |
| 	typename Executor,
 | |
| 	/// Indicates the greatest value the program counter might take.
 | |
| 	uint64_t max_address,
 | |
| 	/// Indicates the maximum number of potential performers that will be provided.
 | |
| 	uint64_t max_performer_count,
 | |
| 	/// Provides the type of Instruction to expect.
 | |
| 	typename InstructionType,
 | |
| 	/// Indicates whether instructions should be treated as ephemeral or included in the cache.
 | |
| 	bool retain_instructions
 | |
| > class CachingExecutor {
 | |
| public:
 | |
| 	using Performer = void (Executor::*)();
 | |
| 	using PerformerIndex = min_int_for_value_t<max_performer_count>;
 | |
| 	using ProgramCounterType = min_int_for_value_t<max_address>;
 | |
| 
 | |
| 	// MARK: - Parser call-ins.
 | |
| 
 | |
| 	void announce_overflow(ProgramCounterType) {
 | |
| 		/*
 | |
| 			Should be impossible for now; this is intended to provide information
 | |
| 			when page caching.
 | |
| 		*/
 | |
| 	}
 | |
| 	void announce_instruction(ProgramCounterType, const InstructionType instruction) {
 | |
| 		// Dutifully map the instruction to a performer and keep it.
 | |
| 		program_.push_back(static_cast<Executor *>(this)->action_for(instruction));
 | |
| 
 | |
| 		if constexpr (retain_instructions) {
 | |
| 			// TODO.
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 
 | |
| 	// Storage for the statically-allocated list of performers. It's a bit more
 | |
| 	// work for executors to fill this array, but subsequently performers can be
 | |
| 	// indexed by array position, which is a lot more compact than a generic pointer.
 | |
| 	std::array<Performer, max_performer_count+1> performers_;
 | |
| 	ProgramCounterType program_counter_;
 | |
| 
 | |
| 	/*!
 | |
| 		Moves the current point of execution to @c address, updating necessary performer caches
 | |
| 		and doing any translation as is necessary.
 | |
| 	*/
 | |
| 	void set_program_counter(ProgramCounterType address) {
 | |
| 		// Set flag to terminate any inner loop currently running through
 | |
| 		// previously-parsed content.
 | |
| 		has_branched_ = true;
 | |
| 		program_counter_ = address;
 | |
| 
 | |
| 		// Temporary implementation: just interpret.
 | |
| 		program_.clear();
 | |
| 		program_index_ = 0;
 | |
| 		static_cast<Executor *>(this)->parse(address, ProgramCounterType(max_address));
 | |
| 
 | |
| //			const auto page = find_page(address);
 | |
| //			const auto entry = page->entry_points.find(address);
 | |
| //			if(entry == page->entry_points.end()) {
 | |
| //				// Requested segment wasn't found; check whether it was
 | |
| //				// within the recently translated list and otherwise
 | |
| //				// translate it.
 | |
| //			}
 | |
| 	}
 | |
| 
 | |
| 	/*!
 | |
| 		Indicates whether the processor is currently 'stopped', i.e. whether all attempts to run
 | |
| 		should produce no activity. Some processors have such a state when waiting for
 | |
| 		interrupts or for a reset.
 | |
| 	*/
 | |
| 	void set_is_stopped(bool) {}
 | |
| 
 | |
| 	/*!
 | |
| 		Executes up to the next branch.
 | |
| 	*/
 | |
| 	void run_to_branch() {
 | |
| 		has_branched_ = false;
 | |
| 		for(auto index: program_) {
 | |
| 			const auto performer = performers_[index];
 | |
| 			(static_cast<Executor *>(this)->*performer)();
 | |
| 			if(has_branched_) break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*!
 | |
| 		Runs for @c duration; the intention is that subclasses provide a method
 | |
| 		that is clear about units, and call this to count down in whatever units they
 | |
| 		count down in.
 | |
| 	*/
 | |
| 	void run_for(int duration) {
 | |
| 		remaining_duration_ += duration;
 | |
| 
 | |
| 		while(remaining_duration_ > 0) {
 | |
| 			has_branched_ = false;
 | |
| 			Executor *const executor = static_cast<Executor *>(this);
 | |
| 			while(remaining_duration_ > 0 && !has_branched_) {
 | |
| 				const auto performer = performers_[program_[program_index_]];
 | |
| 				++program_index_;
 | |
| 
 | |
| 				(executor->*performer)();
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*!
 | |
| 		Should be called by a specific executor to subtract from the remaining
 | |
| 		running duration.
 | |
| 	*/
 | |
| 	inline void subtract_duration(int duration) {
 | |
| 		remaining_duration_ -= duration;
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	bool has_branched_ = false;
 | |
| 	int remaining_duration_ = 0;
 | |
| 	std::vector<PerformerIndex> program_;
 | |
| 	size_t program_index_ = 0;
 | |
| 
 | |
| 	/* TODO: almost below here can be shoved off into an LRUCache object, or similar. */
 | |
| 
 | |
| //		static constexpr size_t max_cached_pages = 64;
 | |
| 
 | |
| //		struct Page {
 | |
| //			std::map<ProgramCounterType, PerformerIndex> entry_points;
 | |
| 
 | |
| 		// TODO: can I statically these two? Should I?
 | |
| //			std::vector<PerformerIndex> actions_;
 | |
| //			std::vector<typename std::enable_if<!std::is_same<InstructionType, void>::value, InstructionType>::type> instructions_;
 | |
| //		};
 | |
| //		std::array<Page, max_cached_pages> pages_;
 | |
| 
 | |
| 	// Maps from page numbers to pages.
 | |
| //		std::unordered_map<ProgramCounterType, Page *> cached_pages_;
 | |
| 
 | |
| 	// Maintains an LRU of recently-used pages in case of a need for reuse.
 | |
| //		std::list<ProgramCounterType> touched_pages_;
 | |
| 
 | |
| 	/*!
 | |
| 		Finds or creates the page that contains @c address.
 | |
| 	*/
 | |
| /*		Page *find_page(ProgramCounterType address) {
 | |
| 		// TODO: are 1kb pages always appropriate? Is 64 the correct amount to keep?
 | |
| 		const auto page_address = ProgramCounterType(address >> 10);
 | |
| 
 | |
| 		auto page = cached_pages_.find(page_address);
 | |
| 		if(page == cached_pages_.end()) {
 | |
| 			// Page wasn't found; either allocate a new one or
 | |
| 			// reuse one that already exists.
 | |
| 			if(cached_pages_.size() == max_cached_pages) {
 | |
| 
 | |
| 			} else {
 | |
| 
 | |
| 			}
 | |
| 		} else {
 | |
| 			// Page was found; LRU shuffle it.
 | |
| 		}
 | |
| 
 | |
| 		return nullptr;
 | |
| 	}*/
 | |
| };
 | |
| 
 | |
| }
 |