blob: 1e39247617ec3e258973f64e9ec6649cca1f590d [file] [log] [blame]
Brian Silverman8efe23e2013-07-07 23:31:37 -07001begin
2 require "sha1"
3rescue LoadError
4 require "digest/sha1"
5end
6
Austin Schuh7e958392014-10-21 22:16:23 -07007$i = 0
8def getForLoopVar()
9 $i = $i + 1
10 return "_autogen_index_#{$i}"
11end
12
brians343bc112013-02-10 01:53:46 +000013module Target
14end
15class Target::Node
16 attr_accessor :created_by
Parker Schuh343481e2014-02-09 18:28:43 -080017 def get_name()
18 if(@parent)
19 return "#{@parent.get_name}.#{@name}"
20 else
21 return "#{@name}"
22 end
23 end
brians343bc112013-02-10 01:53:46 +000024end
25class Target::QFile < Target::Node
26 def initialize() #needs to know repo_path,
27 @class_types = []
28 end
29 def add_type(type)
30 @class_types << type
31 end
32 def extern=(value)
33 @class_types.each do |type|
34 type.extern=value
35 end
36 end
37 def make_cpp_tree(rel_path)
38 cpp_tree = DepFilePair.new(rel_path)
Austin Schuh7e958392014-10-21 22:16:23 -070039 cpp_tree.add_header_include("<array>")
John Park33858a32018-09-28 23:05:48 -070040 cpp_tree.add_header_include("\"aos/macros.h\"")
41 cpp_tree.add_header_include("\"aos/queue.h\"")
brians343bc112013-02-10 01:53:46 +000042 @class_types.each do |type|
43 cpp_tree.get(type)
44 end
45 return cpp_tree
46 end
47end
48class Target::QInclude < Target::Node
49 def initialize(path)
50 @path = path
51 end
Parker Schuhd129f9c2017-02-05 16:07:17 -080052 def extern=(value)
53 @extern=value
54 end
brians343bc112013-02-10 01:53:46 +000055 def create(cpp_tree)
Parker Schuhd129f9c2017-02-05 16:07:17 -080056 return if (@extern)
brians343bc112013-02-10 01:53:46 +000057# inc = cpp_tree.header.add_include("\"#{@path}\"")
58 inc = cpp_tree.add_header_include("\"#{@path}\"")
59 cpp_tree.set(self,inc)
60 end
61end
62class Target::QueueGroupDec < Target::Node
63 attr_accessor :name,:loc,:parent,:queues
64 def initialize(name)
65 @name = name
66 @queues = []
67 @subs = {}
68 end
69 def root()
70 @parent.root()
71 end
72 def []=(key,val)
73 #puts "#{key}= #{val}"
74 @subs[key] = val
75 end
76 def extern=(value)
77 @extern=value
78 end
79 def get_name()
80 if(@parent)
81 return "#{@parent.get_name}.#{@name}"
82 else
83 return "#{@name}"
84 end
85 end
86 def to_cpp_id(id)
87 name = "#{@name}::#{id}"
88 return @parent.to_cpp_id(name) if(@parent)
89 return name
90 end
91 def [](key)
92 #puts "#{key}"
93 @subs[key]
94 end
95 def add_queue(queue)
96 @queues.push(queue)
97 end
98 def hash_with_name(name)
99 ts = (@queues.collect { |queue|
100 queue.msg_hash()
101 }).join("") + name
Brian Silverman8efe23e2013-07-07 23:31:37 -0700102 return "0x#{Digest::SHA1.hexdigest(ts)[-8..-1]}"
brians343bc112013-02-10 01:53:46 +0000103 end
104 def create(cpp_tree)
105 return if(@extern)
106 namespace = cpp_tree.get(@loc)
107 type_class = Types::Class.new(namespace,@name)
108 cpp_tree.set(self,type_class) #breaks infinite recursion
109 type_class.set_parent("public ::aos::QueueGroup")
110 @queues.each do |queue|
111 type_class.add_member(:public,queue.create_usage(cpp_tree))
brians343bc112013-02-10 01:53:46 +0000112 end
113 create_Constructor(type_class,cpp_tree)
114 namespace.add(type_class)
115 end
116 def create_Constructor(type_class,cpp_tree)
117 member_func = CPP::Constructor.new(type_class)
118 type_class.add_member(:public,member_func)
119 #cpp_tree.cc_file.add_funct(member_func)
120 member_func.args << "const char *name";
121 member_func.args << "uint32_t hash";
122 member_func.add_cons("::aos::QueueGroup","name", "hash")
123 @queues.each do |queue|
124 member_func.args << "const char *#{queue.name}_name";
125 member_func.add_cons(queue.name,"#{queue.name}_name")
126 end
127 end
128end
129class Target::QueueGroup < Target::Node
130 attr_accessor :name,:loc
131 def initialize(type,name)
132 @type,@name = type,name
133 end
134 def type_name()
135 if(@type.loc == @loc) #use relative name
136 return @type.name
137 else #use full name
138 return @type.loc.to_cpp_id(@type.name)
139 end
140 end
141 def create(cpp_tree)
142 namespace = cpp_tree.get(@loc)
143 type = cpp_tree.get(@type)
144 cpp_tree.set(self,self)
145 namespace.add_cc("static #{type_name} *#{@name}_ptr")
146 counter = "#{@name}_counter"
147 namespace.add_cc("static int #{counter}")
148
149 str = <<COMMENT_END
150Schwarz counter to construct and destruct the #{@name} object.
151Must be constructed before &#{@name} is constructed so that #{@name}_ptr
152is valid so we can store a reference to the object.
153COMMENT_END
154 str.split(/\n/).each do |str_sec|
155 namespace.class_comment(str_sec)
156 end
157 init_class = namespace.add_class("InitializerFor_" + @name).add_dep(type)
158
159
160 init_class.add_cc_comment(
161 "The counter is initialized at load-time, i.e., before any of the static",
162 "objects are initialized.")
163
164
165 cons = CPP::Constructor.new(init_class)
166 init_class.add_member(:public,cons)
167 cons.suite << if_stmt = CPP::If.new("0 == #{counter}++")
brians343bc112013-02-10 01:53:46 +0000168
169 cons_call = CPP::FuncCall.new("new #{type_name}")
170 cons_call.args.push(@loc.queue_name(@name).inspect)
171
172 cons_call.args.push(@type.hash_with_name(@loc.queue_name(@name).inspect))
173 @type.queues.collect do |queue|
174 cons_call.args.push(@loc.queue_name(@name + "." + queue.name).inspect)
175 end
176 if_stmt.suite << CPP::Assign.new("#{@name}_ptr",cons_call)
brians343bc112013-02-10 01:53:46 +0000177
178
179 destruct = CPP::Destructor.new(init_class)
180 destruct.suite << if_stmt = CPP::If.new("0 == --#{counter}")
brians343bc112013-02-10 01:53:46 +0000181 if_stmt.suite << "delete #{@name}_ptr"
182 if_stmt.suite << CPP::Assign.new("#{@name}_ptr","NULL")
183
184 init_class.add_member(:public,destruct)
185 init_class.static = true
186 init_class.dec = @name + "_initializer";
187
188 str = <<COMMENT_END
189Create a reference to the new object in the pointer. Since we have already
190created the initializer
191COMMENT_END
Brian Silverman04fdc232014-02-12 14:51:11 -0800192 str.split(/\n/).map{|str_sec| CPP::Comment.new(str_sec)}.each do |comment|
193 namespace.add(comment)
194 end
195 namespace.add("static UNUSED_VARIABLE #{type_name} &#{@name}" +
196 " = #{@name}_initializer.get()")
brians343bc112013-02-10 01:53:46 +0000197
198 get = init_class.def_func(type_name,"get") #.add_dep(type)
199 get.pre_func_types = "&"
200 get.suite << CPP::Return.new("*#{@name}_ptr")
201 end
202end