2019-10-04 00:36:49 +03:00
|
|
|
module Sol
|
|
|
|
# This represents a class at the sol level. Sol is a syntax tree,
|
2019-08-08 12:19:27 +03:00
|
|
|
# so here the only child (or children) is a body.
|
|
|
|
# Body may either be a MethodStatement, or Statements (either empty or
|
|
|
|
# containing MethodStatement)
|
|
|
|
#
|
|
|
|
# We store the class name and the parfait class
|
|
|
|
#
|
2019-09-24 15:44:33 +03:00
|
|
|
# The Parfait class gets created by to_parfait, ie only after that is the clazz
|
|
|
|
# attribute set.
|
|
|
|
#
|
2019-08-19 11:33:12 +03:00
|
|
|
class ClassExpression < Expression
|
2017-04-01 15:57:39 +03:00
|
|
|
attr_reader :name, :super_class_name , :body
|
2017-04-08 12:11:52 +03:00
|
|
|
attr_reader :clazz
|
2017-04-01 15:57:39 +03:00
|
|
|
|
|
|
|
def initialize( name , supe , body)
|
2019-09-24 15:44:33 +03:00
|
|
|
@name = name
|
|
|
|
@super_class_name = supe || :Object
|
2019-10-02 17:42:24 +03:00
|
|
|
raise "what body #{body}" unless body.is_a?(Statements)
|
|
|
|
@body = body
|
2017-04-08 12:11:52 +03:00
|
|
|
end
|
|
|
|
|
2019-09-24 15:44:33 +03:00
|
|
|
# This creates the Parfait class.
|
|
|
|
# Creating the class involves creating the instance_type (or an initial version)
|
|
|
|
# which means knowing all used names. So we go through the code looking for
|
|
|
|
# InstanceVariables or InstanceVariable Assignments, to do that.
|
|
|
|
def to_parfait
|
|
|
|
@clazz = Parfait.object_space.get_class_by_name(@name )
|
|
|
|
if(@clazz)
|
|
|
|
if( @super_class_name != clazz.super_class_name)
|
|
|
|
raise "Superclass mismatch for #{@name} , was #{clazz.super_class_name}, now: #{super_class_name}"
|
|
|
|
end
|
|
|
|
else
|
|
|
|
@clazz = Parfait.object_space.create_class(@name , @super_class_name )
|
|
|
|
end
|
|
|
|
create_types
|
|
|
|
@body.statements.each {|meth| meth.to_parfait(@clazz)}
|
|
|
|
@clazz
|
|
|
|
end
|
|
|
|
|
|
|
|
# We transforms every method (class and object)
|
|
|
|
# Other statements are not yet allowed (baring in mind that attribute
|
|
|
|
# accessors are transformed to methods in the ruby layer )
|
2019-08-08 12:19:27 +03:00
|
|
|
#
|
2019-10-03 20:55:41 +03:00
|
|
|
# As there is no class equivalnet in code, a SlotCollection is returned,
|
|
|
|
# which is just a list of SlotMachine::MethodCompilers
|
2019-09-24 15:44:33 +03:00
|
|
|
# The compilers help to transform the code further, into Risc next
|
2019-10-03 20:55:41 +03:00
|
|
|
def to_slot( _ )
|
2018-07-06 20:01:17 +03:00
|
|
|
method_compilers = body.statements.collect do |node|
|
2019-02-17 14:37:50 +02:00
|
|
|
case node
|
2019-08-19 14:33:02 +03:00
|
|
|
when MethodExpression
|
2019-10-03 20:55:41 +03:00
|
|
|
node.to_slot(@clazz)
|
2019-08-19 14:33:02 +03:00
|
|
|
when ClassMethodExpression
|
2019-10-03 20:55:41 +03:00
|
|
|
node.to_slot(@clazz.single_class)
|
2019-02-17 14:37:50 +02:00
|
|
|
else
|
2019-02-16 17:54:45 +02:00
|
|
|
raise "Only methods for now #{node.class}:#{node}"
|
|
|
|
end
|
2018-06-26 20:46:58 +03:00
|
|
|
end
|
2019-10-03 20:55:41 +03:00
|
|
|
SlotMachine::SlotCollection.new(method_compilers)
|
2017-04-12 11:52:23 +03:00
|
|
|
end
|
|
|
|
|
2019-09-24 15:44:33 +03:00
|
|
|
# goes through the code looking for instance variables and their assignments.
|
|
|
|
# Adding each to the respective type, ie class or singleton_class, depending
|
2019-09-19 15:48:27 +03:00
|
|
|
# on if they are instance or class instance variables.
|
|
|
|
#
|
2019-09-19 20:48:21 +03:00
|
|
|
# Class variables are deemed a design mistake, ie not implemented (yet)
|
2019-09-19 15:48:27 +03:00
|
|
|
def create_types
|
|
|
|
self.body.statements.each do |node|
|
|
|
|
case node
|
|
|
|
when MethodExpression
|
|
|
|
target = @clazz
|
|
|
|
when ClassMethodExpression
|
2019-09-24 17:25:19 +03:00
|
|
|
target = @clazz.single_class
|
2019-09-19 15:48:27 +03:00
|
|
|
else
|
|
|
|
raise "Only methods for now #{node.class}:#{node}"
|
|
|
|
end
|
|
|
|
node.each do |exp|
|
|
|
|
case exp
|
|
|
|
when InstanceVariable, IvarAssignment
|
|
|
|
target.add_instance_variable( exp.name , :Object )
|
|
|
|
when ClassVariable #, ClassVarAssignment
|
|
|
|
raise "Class variables not implemented #{node.name}"
|
|
|
|
end
|
|
|
|
end
|
2019-09-18 22:07:05 +03:00
|
|
|
end
|
|
|
|
end
|
2018-07-03 22:18:19 +03:00
|
|
|
def to_s(depth = 0)
|
2019-09-19 20:48:21 +03:00
|
|
|
derive = super_class_name ? "< #{super_class_name}" : ""
|
|
|
|
at_depth(depth , "class #{name} #{derive}\n#{@body.to_s(depth + 1)}\nend")
|
2018-07-03 22:18:19 +03:00
|
|
|
end
|
2017-04-01 15:57:39 +03:00
|
|
|
end
|
|
|
|
end
|