source: asp3_gr_sakura/trunk/tecsgen/tecslib/core/plugin.rb@ 317

Last change on this file since 317 was 317, checked in by coas-nagasima, 7 years ago

GR-SAKURA向けASP3を追加

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-ruby
File size: 12.1 KB
Line 
1# -*- coding: utf-8 -*-
2#
3# TECS Generator
4# Generator for TOPPERS Embedded Component System
5#
6# Copyright (C) 2008-2014 by TOPPERS Project
7#--
8# 上記著作権者
9は,以下の(1)~(4)の条件を満たす場合に限り,本ソフトウェ
10# ア(本ソフトウェアを改変したものを含む.以下同じ)を使用・複製・改
11# 変・再é…
12å¸ƒï¼ˆä»¥ä¸‹ï¼Œåˆ©ç”¨ã¨å‘¼ã¶ï¼‰ã™ã‚‹ã“とを無償で許諾する.
13# (1) 本ソフトウェアをソースコードの形で利用する場合には,上記の著作
14# 権表示,この利用条件および下記の無保証規定が,そのままの形でソー
15# スコード中に含まれていること.
16# (2) 本ソフトウェアを,ライブラリ形式など,他のソフトウェア開発に使
17# 用できる形で再é…
18å¸ƒã™ã‚‹å ´åˆã«ã¯ï¼Œå†é…
19å¸ƒã«ä¼´ã†ãƒ‰ã‚­ãƒ¥ãƒ¡ãƒ³ãƒˆï¼ˆåˆ©ç”¨
20# 者
21マニュアルなど)に,上記の著作権表示,この利用条件および下記
22# の無保証規定を掲載すること.
23# (3) 本ソフトウェアを,機器に組み込むなど,他のソフトウェア開発に使
24# 用できない形で再é…
25å¸ƒã™ã‚‹å ´åˆã«ã¯ï¼Œæ¬¡ã®ã„ずれかの条件を満たすこ
26# と.
27# (a) 再é…
28å¸ƒã«ä¼´ã†ãƒ‰ã‚­ãƒ¥ãƒ¡ãƒ³ãƒˆï¼ˆåˆ©ç”¨è€…
29マニュアルなど)に,上記の著
30# 作権表示,この利用条件および下記の無保証規定を掲載すること.
31# (b) 再é…
32å¸ƒã®å½¢æ…
33‹ã‚’,別に定める方法によって,TOPPERSプロジェクトに
34# 報告すること.
35# (4) 本ソフトウェアの利用により直接的または間接的に生じるいかなる損
36# 害からも,上記著作権者
37およびTOPPERSプロジェクトをå…
38è²¬ã™ã‚‹ã“と.
39# また,本ソフトウェアのユーザまたはエンドユーザからのいかなる理
40# 由に基づく請求からも,上記著作権者
41およびTOPPERSプロジェクトを
42# å…
43è²¬ã™ã‚‹ã“と.
44#
45# 本ソフトウェアは,無保証で提供されているものである.上記著作権者
46お
47# よびTOPPERSプロジェクトは,本ソフトウェアに関して,特定の使用目的
48# に対する適合性も含めて,いかなる保証も行わない.また,本ソフトウェ
49# アの利用により直接的または間接的に生じたいかなる損害に関しても,そ
50# の責任を負わない.
51#
52# $Id: plugin.rb 2061 2014-05-31 22:15:33Z okuma-top $
53#++
54
55#== class モジュール
56# ThroughPlugin, SignaturePlugin, CelltypePlugin に include する
57class Plugin < Node
58#@error_backlog:: [msg1, msg2, ... ] @locale が設定される前に発生したエラー
59
60 PluginArgProc = {
61 "silent" => Proc.new { |obj,rhs| obj.set_silent rhs },
62 }
63
64 def initialize
65 super
66 @b_silent = false
67 @locale = nil # set_locale が呼び出されるまで nil となる
68 @error_backlog = []
69 end
70
71 #=== Plugin#cdl_error
72 # set_locale が呼び出されるまで @error_backlog に保存し保留する
73 def cdl_error *arg
74 if @locale then
75 Generator.error2( @locale, *arg )
76 else
77 @error_backlog << arg
78 end
79 end
80
81 #=== locale を設定する
82 # Node は initialize で locale を設定するが、plugin は parse とは
83 # 異なるタイミング new されるため、locale を再設定する
84 # このメソッドを2度呼び出すと @error_backlog のエラーが2度出力されてしまう
85 def set_locale locale
86 @locale = locale
87 @error_backlog.each { |arg|
88 Generator.error2( locale, *arg )
89 }
90 end
91
92### 意味解析段階で呼び出されるメソッド ###
93 #=== CDL ファイルの生成
94 # typedef, signature, celltype, cell のコードを生成
95 # 重複して生成してはならない
96 # すでに生成されている場合は出力しないこと。
97 # もしくは同名の import により、重複を避けること。
98 #file:: FILE 生成するファイル
99 def gen_cdl_file file
100 end
101
102
103### コード生段階で呼び出されるメソッド ###
104 #=== プラグインは gen_ep_func を提供するか
105 # gen_ep_func 定義 ⇒ テンプレートではない、セルタイプコード(tCelltype.c)を生成
106 # gen_ep_func 未定義 ⇒ テンプレート(tCelltype_templ.c)を生成
107 def gen_ep_func?
108 self.class.method_defined?( :gen_ep_func_body )
109 end
110
111 #=== 受け口関数の本体(C言語)を生成する
112 # プラグインの場合、変更する必
113要のないセルタイプコードを生成する
114 # このメソッドが未定義であれば、プラグインはセルタイプコードを生成しない (通常通りテンプレートを生成する)
115 # gen_cdl_file の中で生成されたセルタイプに対して呼び出される
116 #file:: FILE 出力å…
117ˆãƒ•ã‚¡ã‚¤ãƒ« (tCelltype.c)
118 #b_singleton:: bool true if singleton
119 #ct_name:: Symbol
120 #global_ct_name:: string
121 #sig_name:: string
122 #ep_name:: string
123 #func_name:: string
124 #func_global_name:: string
125 #func_type:: class derived from Type
126# def gen_ep_func_body( file, b_singleton, ct_name, global_ct_name, sig_name, ep_name, func_name, func_global_name, func_type, params )
127# end
128
129 #=== 受け口関数の preamble (C言語)を生成する
130 # 必
131要なら preamble 部に出力する
132 # gen_cdl_file の中でで生成されたセルタイプに対して呼び出される
133 #file:: FILE 出力å…
134ˆãƒ•ã‚¡ã‚¤ãƒ«
135 #b_singleton:: bool true if singleton
136 #ct_name:: Symbol
137 #global_ct_name:: string
138 def gen_preamble( file, b_singleton, ct_name, global_ct_name )
139 # デフォルトでは何も出力しない
140 end
141
142 #=== 受け口関数の postamble (C言語)を生成する
143 # 必
144要なら postamble 部に出力する
145 # gen_cdl_file の中で生成されたセルタイプに対して呼び出される
146 #file:: FILE 出力å…
147ˆãƒ•ã‚¡ã‚¤ãƒ«
148 #b_singleton:: bool true if singleton
149 #ct_name:: Symbol
150 #global_ct_name:: string
151 def gen_postamble( file, b_singleton, ct_name, global_ct_name )
152 # デフォルトでは何も出力しない
153 end
154
155 #=== gen_cdl_file の中で生成されたセルタイプに新しいセルが生成された
156 # どのセルタイプかは cell.get_celltype で分かる
157 #
158 #file:: FILE 出力å…
159ˆãƒ•ã‚¡ã‚¤ãƒ«
160 #b_singleton:: bool true if singleton
161 #ct_name:: Symbol
162 #global_ct_name:: string
163 def new_cell cell
164 # デフォルトでは何もしない
165 end
166
167### プラグイン引数の解釈 ###
168 def parse_plugin_arg
169 arg = @plugin_arg_str.dup
170
171 # 改行を消す
172 arg.gsub!( /\\\n/, "" )
173
174 while arg != ""
175
176 # 前の空白読み飛ばす
177 arg.sub!( /\A\s*(?:\\\n)*\s*(.*)/, '\1')
178
179 # 識別子取得
180 if arg =~ /\A[a-zA-Z_]\w*/ then
181 ident = $~
182 arg = $'
183 else
184 cdl_error( "P1001 plugin arg: cannot find identifier in $1" , arg )
185 return
186 end
187
188 # 前の空白読み飛ばす
189 arg.sub!( /\A\s*(?:\\\n)*\s*(.*)/, '\1')
190
191 if arg =~ /=/ then
192 arg = $'
193 else
194 cdl_error( "P1002 plugin arg: expecting \'=\' not \'$1\'" , arg )
195 return
196 end
197
198 # 前の空白読み飛ばす
199 arg.sub!( /\A\s*(?:\\\n)*\s*(.*)/, '\1')
200
201 # 右辺文字列
202 if arg =~ /\A\\"(.*?)\\"\s*,/ then # \" \" で囲まれている場合
203 rhs = $1
204 remain = $'
205 elsif arg =~ /\A%(.*?)%\s*,/ then # % % で囲まれている場合
206 rhs = $1
207 remain = $'
208 elsif arg =~ /\A!(.*?)!\s*,/ then # $ $ で囲まれている場合
209 rhs = $1
210 remain = $'
211 elsif arg =~ /\A'(.*?)'\s*,/ then # $ $ で囲まれている場合
212 rhs = $1
213 remain = $'
214 elsif arg =~ /\A\\"(.*?)\\"\s*,/ then # || にも [,$] にもできなかった
215 rhs = $1
216 remain = $'
217 # elsif arg =~ /\A(.*?)\s*$/ then
218 elsif arg =~ /\A\\"(.*?)\\"\s*\z/ then # \" \" で囲まれている場合
219 rhs = $1
220 remain = $'
221 elsif arg =~ /\A%(.*?)%\s*\z/ then # % % で囲まれている場合
222 rhs = $1
223 remain = $'
224 elsif arg =~ /\A!(.*?)!\s*\z/ then # $ $ で囲まれている場合
225 rhs = $1
226 remain = $'
227 elsif arg =~ /\A'(.*?)'\s*\z/ then # $ $ で囲まれている場合
228 rhs = $1
229 remain = $'
230 elsif arg =~ /\A\\"(.*?)\\"\s*\z/ then # || にも [,$] にもできなかった
231 rhs = $1
232 remain = $'
233 elsif arg =~ /\A(.*?),/ then
234 rhs = $1
235 remain = $'
236 # 前の空白読み飛ばす
237 rhs.sub!( /\A\s*(.*)\s*\z/, '\1')
238 elsif arg =~ /\A(.*?)\s*\z/ then
239 rhs = $1
240 remain = $'
241 else
242 cdl_error( "P1003 plugin arg: unexpected $1" , arg )
243 return
244 end
245
246 # 0文字の文字列を to_sym すると例外発生するので空白文字とする
247 if rhs == "" then
248 rhs = " "
249 end
250
251 arg = remain # arg の残りの部分
252 arg.sub!( /\A\s*(?:\\\n)*\s*(.*)/, '\1') # 前の空白読み飛ばす
253
254 # \ を外す
255 rhs = rhs.gsub( /\\(.)/, "\\1" ) # ここで $' が変わることに注意!
256 # print "parse_plugin_arg: #{ident} #{rhs}\n"
257 @plugin_arg_list[ ident ] = rhs
258
259 check_plugin_arg( ident, rhs )
260 end
261 # @plugin_arg_list.each{|i,r| print "ident: #{i} rhs: #{r}\n" }
262 end
263
264 #=== プラグイン引数をチェックする
265 # @plugin_arg_check_proc_tab に従ってプラグイン引数をチェックすする
266 # 古い用法:子クラスでオーバーライドし、引数識別子が正しいかチェックする
267 #ident:: string: 引数識別子
268 #rhs:: string: 右辺文字列
269 def check_plugin_arg( ident, rhs )
270
271 dbgPrint "check_plugin_arg: #{ident} #{rhs.to_str}\n"
272 proc = nil
273 if @plugin_arg_check_proc_tab then
274 proc = @plugin_arg_check_proc_tab[ident.to_s]
275 end
276 if proc == nil then
277 proc = PluginArgProc[ ident.to_s ]
278 end
279 if proc.instance_of? Proc then
280 dbgPrint "calling: #{self.class.name}.#{proc.to_s}\n"
281 proc.call( self, rhs )
282 else
283 params = ""
284 delim = ""
285 @plugin_arg_check_proc_tab.each{ |j,p|
286 params = "#{params}#{delim}#{j}"
287 delim = ", "
288 }
289 cdl_error( "P1004 $1: unknown plugin argument\'s identifier\n $2 are acceptible for RPCPlugin." , ident, params )
290 end
291 end
292
293 #=== プラグインのメッセージ出力
294 def print_msg( msg )
295 if @b_silent == true then
296 return
297 end
298 print msg
299 end
300
301 #=== プラグイン引数 silent
302 def set_silent rhs
303 if rhs == "true" || rhs == nil then
304 @b_silent = true
305 end
306 end
307end
308
309#== 出力文字列を EUC から出力ファイルに convert する
310# tecsgen のソースコードは EUC で記述されている
311# これを、出力ファイルの文字コードに変換して出力する
312#
313# generate.rb で出力するものは message.rb で変換している
314# generate.rb で出力するものは APPFile クラスを使用している
315# mikan: CFile で出力したものに factory で追記できない (cdl ファイルの場合、追記できても意味がない)
316class CFile
317 def self.open( path, mode )
318 CFile.new( path, mode )
319 end
320
321 def initialize( path, mode )
322 if $b_no_kcode then
323 mode += ":" + $Ruby19_File_Encode
324 end
325 @file = File.open( path, mode )
326 end
327
328 def print str
329 if $b_no_kcode && $KCONV_CONSOLE == Kconv::BINARY then
330 @file.print( str )
331 else
332 @file.print( str.kconv( $KCONV_CDL, $KCONV_TECSGEN ) )
333 end
334 end
335
336 def puts str
337 if $b_no_kcode && $KCONV_CONSOLE == Kconv::BINARY then
338 @file.print( str )
339 else
340 @file.print( str.kconv( $KCONV_CDL, $KCONV_TECSGEN ) )
341 end
342 @file.print( "\n" )
343 end
344
345 def printf( format, *arg )
346 if $b_no_kcode && $KCONV_CONSOLE == Kconv::BINARY then
347 @file.print( sprintf( format, *arg ) )
348 else
349 @file.print( sprintf( format, *arg ).kconv( $KCONV_CDL, $KCONV_TECSGEN ) )
350 end
351 end
352
353 def close
354 @file.close
355 end
356end
Note: See TracBrowser for help on using the repository browser.