require 'pry-byebug' module Assembler6502 class SymbolTable attr_accessor :scope_stack ##### Custom Exceptions class InvalidScope < StandardError; end class UndefinedSymbol < StandardError; end class CantExitScope < StandardError; end #### ## Initialize a symbol table that begins in global scope def initialize @symbols = { :global => {} } @anonymous_scope_number = 0 @scope_stack = [:global] end #### ## Define a new scope, which can be anonymous or named ## and switch into that scope def enter_scope(name = nil) name = generate_name if name.nil? name = name.to_sym scope = current_scope if scope.has_key?(name) #path_string = generate_scope_path(path_ary) fail(InvalidScope, "Scope: #{name} already exists") end scope[name] = {} @scope_stack.push(name) end #### ## Exit the current scope def exit_scope if @scope_stack.size == 1 fail(CantExitScope, "You cannot exit global scope") end @scope_stack.pop end #### ## Define a symbol in the current scope def define_symbol(symbol, value) scope = current_scope scope[symbol.to_sym] = value end =begin #### ## Resolve symbol to a value, for example: ## scope1.scope2.variable ## It is not nessessary to specify the root scope :global ## You can just address anything by name in the current scope ## To go backwards in scope you need to write the full path ## like global.sprite.x or whatever def resolve_symbol_old(name) value = if name.include?('.') path_ary = name.split('.').map(&:to_sym) symbol = path_ary.pop path_ary.shift if path_ary.first == :global scope = retreive_scope(path_ary) ## We also try to look up the address associated with the scope root = "-#{symbol}".to_sym v = scope[symbol] v.kind_of?(Hash) ? v[root] : v else root = "-#{name}".to_sym scope = current_scope ## We also try to look up the address associated with the scope v = scope[name.to_sym] || scope[root] v.kind_of?(Hash) ? v[root] : v end if value.nil? fail(UndefinedSymbol, name) end value end =end #### ## def resolve_symbol(name) method = name.include?('.') ? :resolve_symbol_dot_syntax : :resolve_symbol_scoped value = self.send(method, name) fail(UndefinedSymbol, name) if value.nil? value end #### ## Resolve symbol by working backwards through each ## containing scope. Similarly named scopes shadow outer scopes def resolve_symbol_scoped(name) root = "-#{name}".to_sym stack = @scope_stack.dup loop do scope = retreive_scope(stack) ## We see if there is a key either under this name, or root v = scope[name.to_sym] || scope[root] v = v.kind_of?(Hash) ? v[root] : v return v unless v.nil? ## Pop the stack so we can decend to the parent scope, if any stack.pop return nil if stack.empty? end end #### ## Dot syntax means to check an absolute path to the symbol ## :global is ignored if it is provided as part of the path def resolve_symbol_dot_syntax(name) path_ary = name.split('.').map(&:to_sym) symbol = path_ary.pop root = "-#{symbol}".to_sym path_ary.shift if path_ary.first == :global scope = retreive_scope(path_ary) ## We see if there is a key either under this name, or root v = scope[symbol] v.kind_of?(Hash) ? v[root] : v end #### ## Export the symbol table as YAML def export_to_yaml @symbols.to_yaml.gsub(/(\d+)$/) do |match| integer = match.to_i sprintf("0x%.4X", integer) end end private #### ## A bit more clearly states to get the current scope def current_scope retreive_scope end #### ## Retrieve a reference to a scope, current scope by default def retreive_scope(path_ary = @scope_stack) path_ary = path_ary.dup path_ary.unshift(:global) unless path_ary.first == :global path_ary.inject(@symbols) do |scope, path_component| new_scope = scope[path_component.to_sym] if new_scope.nil? path_string = generate_scope_path(path_ary) message = "Resolving scope: #{path_string} failed at #{path_component}" fail(InvalidScope, message) if new_scope.nil? end new_scope end end #### ## Generate a scope path from an array def generate_scope_path(path_ary) path_ary.join('.') end #### ## Generate an anonymous scope name def generate_name @anonymous_scope_number += 1 "anonymous_#{@anonymous_scope_number}" end end end