source: asp3_wo_tecs/trunk/extension/ovrhdr/kernel/kernel.trb@ 306

Last change on this file since 306 was 306, checked in by ertl-honda, 7 years ago

3.1.0を反映

File size: 15.3 KB
Line 
1# -*- coding: utf-8 -*-
2#
3# TOPPERS/ASP Kernel
4# Toyohashi Open Platform for Embedded Real-Time Systems/
5# Advanced Standard Profile Kernel
6#
7# Copyright (C) 2015 by FUJI SOFT INCORPORATED, JAPAN
8# Copyright (C) 2015,2016 by Embedded and Real-Time Systems Laboratory
9# Graduate School of Information Science, Nagoya Univ., JAPAN
10#
11# 上記著作権者
12は,以下の(1)〜(4)の条件を満たす場合に限り,本ソフトウェ
13# ア(本ソフトウェアを改変したものを含む.以下同じ)を使用・複製・改
14# 変・再é…
15å¸ƒï¼ˆä»¥ä¸‹ï¼Œåˆ©ç”¨ã¨å‘¼ã¶ï¼‰ã™ã‚‹ã“とを無償で許諾する.
16# (1) 本ソフトウェアをソースコードの形で利用する場合には,上記の著作
17# 権表示,この利用条件および下記の無保証規定が,そのままの形でソー
18# スコード中に含まれていること.
19# (2) 本ソフトウェアを,ライブラリ形式など,他のソフトウェア開発に使
20# 用できる形で再é…
21å¸ƒã™ã‚‹å ´åˆã«ã¯ï¼Œå†é…
22å¸ƒã«ä¼´ã†ãƒ‰ã‚­ãƒ¥ãƒ¡ãƒ³ãƒˆï¼ˆåˆ©ç”¨
23# 者
24マニュアルなど)に,上記の著作権表示,この利用条件および下記
25# の無保証規定を掲載すること.
26# (3) 本ソフトウェアを,機器に組み込むなど,他のソフトウェア開発に使
27# 用できない形で再é…
28å¸ƒã™ã‚‹å ´åˆã«ã¯ï¼Œæ¬¡ã®ã„ずれかの条件を満たすこ
29# と.
30# (a) 再é…
31å¸ƒã«ä¼´ã†ãƒ‰ã‚­ãƒ¥ãƒ¡ãƒ³ãƒˆï¼ˆåˆ©ç”¨è€…
32マニュアルなど)に,上記の著
33# 作権表示,この利用条件および下記の無保証規定を掲載すること.
34# (b) 再é…
35å¸ƒã®å½¢æ…
36‹ã‚’,別に定める方法によって,TOPPERSプロジェクトに
37# 報告すること.
38# (4) 本ソフトウェアの利用により直接的または間接的に生じるいかなる損
39# 害からも,上記著作権者
40およびTOPPERSプロジェクトをå…
41è²¬ã™ã‚‹ã“と.
42# また,本ソフトウェアのユーザまたはエンドユーザからのいかなる理
43# 由に基づく請求からも,上記著作権者
44およびTOPPERSプロジェクトを
45# å…
46è²¬ã™ã‚‹ã“と.
47#
48# 本ソフトウェアは,無保証で提供されているものである.上記著作権者
49お
50# よびTOPPERSプロジェクトは,本ソフトウェアに関して,特定の使用目的
51# に対する適合性も含めて,いかなる保証も行わない.また,本ソフトウェ
52# アの利用により直接的または間接的に生じたいかなる損害に関しても,そ
53# の責任を負わない.
54#
55# $Id: kernel.trb 670 2016-03-03 15:08:11Z ertl-hiro $
56#
57
58#
59# コンフィギュレータのパス2の生成スクリプト
60#
61
62#
63# タイムスタンプファイルの指定
64#
65$timeStampFileName = "kernel_cfg.timestamp"
66
67#
68# kernel_cfg.hのå…
69ˆé ­éƒ¨åˆ†ã®ç”Ÿæˆ
70#
71$kernelCfgH = GenFile.new("kernel_cfg.h")
72$kernelCfgH.add(<<EOS)
73/* kernel_cfg.h */
74#ifndef TOPPERS_KERNEL_CFG_H
75#define TOPPERS_KERNEL_CFG_H
76EOS
77
78#
79# kernel_cfg.cのå…
80ˆé ­éƒ¨åˆ†ã®ç”Ÿæˆ
81#
82$kernelCfgC = GenFile.new("kernel_cfg.c")
83$kernelCfgC.add(<<EOS)
84/* kernel_cfg.c */
85#include "kernel/kernel_int.h"
86#include "kernel_cfg.h"
87
88#if !(TKERNEL_PRID == 0x0007U && (TKERNEL_PRVER & 0xf000U) == 0x3000U)
89#error The kernel does not match this configuration file.
90#endif
91EOS
92
93#
94# インクルードディレクティブ(#include)
95#
96$kernelCfgC.comment_header("Include Directives")
97$includeFiles.each do |file|
98 $kernelCfgC.add("#include #{file}")
99end
100$kernelCfgC.add()
101
102#
103# スタック領域の確保関数
104#
105# スタック領域の定義分と,スタック領域のサイズ記述をé…
106åˆ—で返す.
107#
108unless defined? AllocStack()
109 def AllocStack(stack, size)
110 # 大きい方に丸めたサイズで確保する[NGKI1050][NGKI3220]
111 $kernelCfgC.add("static STK_T #{stack}[COUNT_STK_T(#{size})];")
112 return("ROUND_STK_T(#{size})")
113 end
114end
115
116#
117# カーネルオブジェクトに関する情
118報の生成(仮想クラス)
119#
120class KernelObject
121 def initialize(obj, object, obj_s = obj)
122 @obj = obj
123 @OBJ = obj.tr("a-z", "A-Z")
124 @object = object
125 @obj_s = obj_s
126 @OBJ_S = obj_s.tr("a-z", "A-Z")
127 @objid = (obj + "id").to_sym
128 @api = "CRE_#{@OBJ}".to_sym
129 end
130
131 def generate()
132 # kernel_cfg.hの生成
133 $kernelCfgH.add("#define TNUM_#{@OBJ}ID\t#{$cfgData[@api].size}")
134
135 # オブジェクトのID番号のマクロ定義の生成
136 $cfgData[@api].sort.each do |key, params|
137 $kernelCfgH.add("#define #{params[@objid]}\t#{params[@objid].val}")
138 end
139 $kernelCfgH.add()
140
141 # オブジェクトのID番号を保持する変数
142 if $USE_EXTERNAL_ID
143 $cfgData[@api].sort.each do |key, params|
144 $kernelCfgC.add("const ID #{params[@objid]}_id" \
145 " = #{params[@objid].val};")
146 end
147 $kernelCfgC.add()
148 end
149
150 # オブジェクトID番号の最大値
151 $kernelCfgC.add2("const ID _kernel_tmax_#{@obj}id" \
152 " = (TMIN_#{@OBJ}ID + TNUM_#{@OBJ}ID - 1);")
153
154 # データ構造
155 if $cfgData[@api].size > 0
156 # 事前準備(エラーチェック,メモリ領域の生成)
157 $cfgData[@api].sort.each do |key, params|
158 prepare(key, params)
159 end
160
161 # オブジェクトå…
162¨ä½“に対して必
163要なメモリ領域の生成(オプション)
164 if respond_to?(:generateData)
165 generateData()
166 end
167
168 # オブジェクト初期化ブロックの生成
169 $kernelCfgC.add("const #{@OBJ_S}INIB _kernel_#{@obj_s}inib_table" \
170 "[TNUM_#{@OBJ}ID] = {")
171 $cfgData[@api].sort.each_with_index do |(key, params), index|
172 $kernelCfgC.add(",") if index > 0
173 $kernelCfgC.append("\t{ " + generateInib(key, params) + " }")
174 end
175 $kernelCfgC.add
176 $kernelCfgC.add2("};")
177
178 # オブジェクト管理ブロックの生成
179 $kernelCfgC.add2("#{@OBJ_S}CB _kernel_#{@obj_s}cb_table" \
180 "[TNUM_#{@OBJ}ID];")
181
182 # オブジェクト初期化関数の追加
183 $initializeFunctions.push("_kernel_initialize_#{@object}();")
184 else
185 # オブジェクトが1つもない場合
186 $kernelCfgC.add("TOPPERS_EMPTY_LABEL(const #{@OBJ_S}INIB, " \
187 "_kernel_#{@obj_s}inib_table);")
188 $kernelCfgC.add2("TOPPERS_EMPTY_LABEL(#{@OBJ_S}CB, " \
189 "_kernel_#{@obj_s}cb_table);")
190 end
191 end
192end
193
194#
195# 通知ハンドラの生成関数
196#
197def generateNotifyHandler(key, params, objid)
198 # パラメータを変数に格納
199 nfymode = params[:nfymode]
200 nfymode1 = nfymode & 0x0f
201 nfymode2 = nfymode & ~0x0f
202 par1 = params[:par1]
203 par2 = params[:par2]
204
205 # 通知処理のパラメータ数による補正処理
206 if nfymode == $TNFY_HANDLER || nfymode1 == $TNFY_SETVAR \
207 || nfymode1 == $TNFY_SETFLG \
208 || nfymode1 == $TNFY_SNDDTQ
209 # 通知処理のパラメータが2つの場合
210 numpar = 2
211 epar1 = params[:par3]
212 epar2 = params[:par4]
213 else
214 # 通知処理のパラメータが1つの場合
215 numpar = 1
216 epar1 = params[:par2]
217 epar2 = params[:par3]
218 end
219
220 # パラメータ数のチェック
221 if (numpar == 2 && par2.nil?) || (nfymode2 != 0 && epar1.nil?) \
222 || (nfymode2 == $TENFY_SETFLG && epar2.nil?)
223 # パラメータが足りない場合
224 error_api(params, "too few parameters for nfymode `#{nfymode}' " \
225 "in %apiname of %#{objid}")
226 elsif (nfymode2 == 0 && !epar1.nil?) \
227 || (nfymode2 != $TENFY_SETFLG && !epar2.nil?)
228 # パラメータが多すぎる場合
229 error_api(params, "too many parameters for nfymode `#{nfymode}' " \
230 "in %apiname of %#{objid}")
231 elsif nfymode == $TNFY_HANDLER
232 # タイムイベントハンドラの呼出し
233 funcname = "(NFYHDR)(#{par2})"
234 else
235 # 通知ハンドラの関数名
236 funcname = "_kernel_nfyhdr_#{params[objid]}"
237
238 # エラー通知のための変数のアドレスとオブジェクトIDを格納する
239 # 変数の生成(エラーチェックのために必
240要)
241 if nfymode2 == $TENFY_SETVAR || nfymode2 == $TENFY_INCVAR
242 $kernelCfgC.add2("intptr_t *const #{funcname}_p_evar =" \
243 " (intptr_t *)(#{epar1});")
244 elsif nfymode2 == $TENFY_ACTTSK || nfymode2 == $TENFY_WUPTSK
245 $kernelCfgC.add2("const ID #{funcname}_etskid = #{epar1};")
246 elsif nfymode2 == $TENFY_SIGSEM
247 $kernelCfgC.add2("const ID #{funcname}_esemid = #{epar1};")
248 elsif nfymode2 == $TENFY_SETFLG
249 $kernelCfgC.add2("const ID #{funcname}_eflgid = #{epar1};")
250 elsif nfymode2 == $TENFY_SNDDTQ
251 $kernelCfgC.add2("const ID #{funcname}_edtqid = #{epar1};")
252 end
253
254 # 関数のå…
255ˆé ­éƒ¨åˆ†ã®ç”Ÿæˆ
256 $kernelCfgC.add("static void")
257 $kernelCfgC.add("#{funcname}(intptr_t exinf)")
258 $kernelCfgC.add("{")
259
260 if nfymode2 == 0
261 # エラー通知がない場合
262 errorCode = "(void) "
263 else
264 # エラー通知がある場合
265 $kernelCfgC.add2("\tER\tercd;")
266 errorCode = "ercd = "
267 end
268
269 # イベント通知処理の処理
270 if nfymode1 == $TNFY_SETVAR && nfymode2 == 0
271 # 変数の設定
272 $kernelCfgC.add("\t*((intptr_t *) exinf) = (#{par2});")
273 elsif nfymode1 == $TNFY_INCVAR && nfymode2 == 0
274 # 変数のインクリメント
275 $kernelCfgC.add("\t(void) loc_cpu();")
276 $kernelCfgC.add("\t*((intptr_t *) exinf) += 1;")
277 $kernelCfgC.add("\t(void) unl_cpu();")
278 elsif nfymode1 == $TNFY_ACTTSK
279 # タスクの起動
280 $kernelCfgC.add("\t#{errorCode}act_tsk((ID) exinf);")
281 elsif nfymode1 == $TNFY_WUPTSK
282 # タスクの起床
283 $kernelCfgC.add("\t#{errorCode}wup_tsk((ID) exinf);")
284 elsif nfymode1 == $TNFY_SIGSEM
285 # セマフォの返却
286 $kernelCfgC.add("\t#{errorCode}sig_sem((ID) exinf);")
287 elsif nfymode1 == $TNFY_SETFLG
288 # イベントフラグのセット
289 $kernelCfgC.add("\t#{errorCode}set_flg(((ID) exinf), #{par2});")
290 elsif nfymode1 == $TNFY_SNDDTQ
291 # データキューへの送信
292 $kernelCfgC.add("\t#{errorCode}psnd_dtq(((ID) exinf), #{par2});")
293 else
294 error_illegal_id("E_PAR", params, :nfymode, objid)
295 end
296
297 if nfymode2 != 0
298 # エラー通知処理の処理
299 $kernelCfgC.add("\tif (ercd != E_OK) {")
300
301
302 if nfymode2 == $TENFY_SETVAR
303 # 変数の設定
304 $kernelCfgC.add("\t\t*#{funcname}_p_evar = (intptr_t) ercd;")
305 elsif nfymode2 == $TENFY_INCVAR
306 # 変数のインクリメント
307 $kernelCfgC.add("\t\t(void) loc_cpu();")
308 $kernelCfgC.add("\t\t*#{funcname}_p_evar += 1;")
309 $kernelCfgC.add("\t\t(void) unl_cpu();")
310 elsif nfymode2 == $TENFY_ACTTSK
311 # タスクの起動
312 $kernelCfgC.add("\t\t(void) act_tsk(#{funcname}_etskid);")
313 elsif nfymode2 == $TENFY_WUPTSK
314 # タスクの起床
315 $kernelCfgC.add("\t\t(void) wup_tsk(#{funcname}_etskid);")
316 elsif nfymode2 == $TENFY_SIGSEM
317 # セマフォの返却
318 $kernelCfgC.add("\t\t(void) sig_sem(#{funcname}_esemid);")
319 elsif nfymode2 == $TENFY_SETFLG
320 # イベントフラグのセット
321 $kernelCfgC.add("\t\t(void) set_flg(#{funcname}_eflgid, #{epar2});")
322 elsif nfymode2 == $TENFY_SNDDTQ
323 # データキューへの送信
324 $kernelCfgC.add("\t\t(void) psnd_dtq(#{funcname}_edtqid," \
325 " (intptr_t) ercd);")
326 else
327 error_illegal_id("E_PAR", params, :nfymode, objid)
328 end
329 $kernelCfgC.add("\t}")
330 end
331
332 # 関数の末尾部分の生成
333 $kernelCfgC.add2("}")
334 end
335 return(funcname)
336end
337
338#
339# 各機能モジュールのコンフィギュレーション
340#
341$initializeFunctions = []
342IncludeTrb("kernel/task.trb")
343IncludeTrb("kernel/semaphore.trb")
344IncludeTrb("kernel/eventflag.trb")
345IncludeTrb("kernel/dataqueue.trb")
346IncludeTrb("kernel/pridataq.trb")
347IncludeTrb("kernel/mutex.trb")
348IncludeTrb("kernel/mempfix.trb")
349IncludeTrb("kernel/cyclic.trb")
350IncludeTrb("kernel/alarm.trb")
351IncludeTrb("kernel/overrun.trb")
352IncludeTrb("kernel/interrupt.trb")
353IncludeTrb("kernel/exception.trb")
354
355#
356# 非タスクコンテキスト用のスタック領域
357#
358$kernelCfgC.comment_header("Stack Area for Non-task Context")
359
360if $cfgData[:DEF_ICS].size == 0
361 # DEF_ICSがない場合のデフォルト値の設定
362 if $DEFAULT_ISTK.nil?
363 # スタック領域の自動割付け
364 istksz = AllocStack("_kernel_istack", "DEFAULT_ISTKSZ")
365 istk = "_kernel_istack"
366 else
367 istksz = "DEFAULT_ISTKSZ"
368 istk = "DEFAULT_ISTK"
369 end
370else
371 # 静的API「DEF_ICS」が複数ある(E_OBJ)[NGKI3216]
372 if $cfgData[:DEF_ICS].size > 1
373 error("E_OBJ: too many DEF_ICS")
374 end
375
376 # DEF_ICSがある場合の処理
377 params = $cfgData[:DEF_ICS][1]
378
379 # パラメータが省略された時のデフォルト値の設定
380 params[:istk] ||= "NULL"
381
382 # istkszがターゲット定義の最小値(TARGET_MIN_ISTKSZ,未定義の場合は1)
383 # よりも小さい場合(E_PAR)[NGKI3254]
384 if params[:istksz] < $TARGET_MIN_ISTKSZ
385 error_wrong("E_PAR", params, :istksz, "too small")
386 end
387
388 if params[:istk] == "NULL"
389 # スタック領域の自動割付け
390 istksz = AllocStack("_kernel_istack", params[:istksz])
391 istk = "_kernel_istack"
392 else
393 # istkszがスタック領域のサイズとして正しくない場合(E_PAR)[NGKI3222]
394 if (params[:istksz] & ($CHECK_STKSZ_ALIGN - 1)) != 0
395 error_wrong("E_PAR", params, :istksz, "not aligned")
396 end
397
398 istksz = "(#{params[:istksz]})"
399 istk = "(void *)(#{params[:istk]})"
400 end
401end
402
403$kernelCfgC.add(<<EOS)
404const size_t _kernel_istksz = #{istksz};
405STK_T *const _kernel_istk = #{istk};
406
407#ifdef TOPPERS_ISTKPT
408STK_T *const _kernel_istkpt = TOPPERS_ISTKPT(#{istk}, #{istksz});
409#endif /* TOPPERS_ISTKPT */
410EOS
411
412#
413# タイムイベント管理
414#
415$kernelCfgC.comment_header("Time Event Management")
416$kernelCfgC.add(<<EOS)
417TMEVTN _kernel_tmevt_heap[1 + TNUM_TSKID + TNUM_CYCID + TNUM_ALMID];
418EOS
419
420#
421# 各モジュールの初期化関数
422#
423$kernelCfgC.comment_header("Module Initialization Function")
424$kernelCfgC.append(<<EOS)
425void
426_kernel_initialize_object(void)
427{
428EOS
429$initializeFunctions.each do |func|
430 $kernelCfgC.add("\t#{func}")
431end
432$kernelCfgC.add2("}")
433
434#
435# 初期化ルーチン機能
436#
437$kernelCfgC.comment_header("Initialization Routine")
438
439# エラーチェック
440$cfgData[:ATT_INI].each do |key, params|
441 # iniatrが無効の場合(E_RSATR)[NGKI3241]
442 #(TA_NULLでない場合)
443 if (params[:iniatr] != $TA_NULL)
444 error_illegal_sym("E_RSATR", params, :iniatr, :inirtn)
445 end
446end
447
448# 初期化ルーチンの実行関数の生成
449$kernelCfgC.append(<<EOS)
450void
451_kernel_call_inirtn(void)
452{
453EOS
454$cfgData[:ATT_INI].each do |key, params|
455 $kernelCfgC.add("\t((INIRTN)(#{params[:inirtn]}))" \
456 "((intptr_t)(#{params[:exinf]}));")
457end
458$kernelCfgC.add2("}")
459
460#
461# 終了処理ルーチン機能
462#
463$kernelCfgC.comment_header("Termination Routine")
464
465# エラーチェック
466$cfgData[:ATT_TER].each do |key, params|
467 # teratrが無効の場合(E_RSATR)[NGKI3248]
468 #(TA_NULLでない場合)
469 if (params[:teratr] != $TA_NULL)
470 error_illegal_sym("E_RSATR", params, :teratr, :terrtn)
471 end
472end
473
474# 終了処理ルーチンの実行関数の生成
475$kernelCfgC.append(<<EOS)
476void
477_kernel_call_terrtn(void)
478{
479EOS
480$cfgData[:ATT_TER].reverse_each do |key, params|
481 $kernelCfgC.add("\t((TERRTN)(#{params[:terrtn]}))" \
482 "((intptr_t)(#{params[:exinf]}));")
483end
484$kernelCfgC.add2("}")
485
486#
487# kernel_cfg.hの末尾部分の生成
488#
489$kernelCfgH.append(<<EOS)
490#endif /* TOPPERS_KERNEL_CFG_H */
491EOS
Note: See TracBrowser for help on using the repository browser.