1 | /*
|
---|
2 | * TOPPERS ECHONET Lite Communication Middleware
|
---|
3 | *
|
---|
4 | * Copyright (C) 2014-2016 Cores Co., Ltd. Japan
|
---|
5 | *
|
---|
6 | * 上記著作権者は,以下の(1)〜(4)の条件を満たす場合に限り,本ソフトウェ
|
---|
7 | * ア(本ソフトウェアを改変したものを含む.以下同じ)を使用・複製・改
|
---|
8 | * 変・再配布(以下,利用と呼ぶ)することを無償で許諾する.
|
---|
9 | * (1) 本ソフトウェアをソースコードの形で利用する場合には,上記の著作
|
---|
10 | * 権表示,この利用条件および下記の無保証規定が,そのままの形でソー
|
---|
11 | * スコード中に含まれていること.
|
---|
12 | * (2) 本ソフトウェアを,ライブラリ形式など,他のソフトウェア開発に使
|
---|
13 | * 用できる形で再配布する場合には,再配布に伴うドキュメント(利用
|
---|
14 | * 者マニュアルなど)に,上記の著作権表示,この利用条件および下記
|
---|
15 | * の無保証規定を掲載すること.
|
---|
16 | * (3) 本ソフトウェアを,機器に組み込むなど,他のソフトウェア開発に使
|
---|
17 | * 用できない形で再配布する場合には,次のいずれかの条件を満たすこ
|
---|
18 | * と.
|
---|
19 | * (a) 再配布に伴うドキュメント(利用者マニュアルなど)に,上記の著
|
---|
20 | * 作権表示,この利用条件および下記の無保証規定を掲載すること.
|
---|
21 | * (b) 再配布の形態を,別に定める方法によって,TOPPERSプロジェクトに
|
---|
22 | * 報告すること.
|
---|
23 | * (4) 本ソフトウェアの利用により直接的または間接的に生じるいかなる損
|
---|
24 | * 害からも,上記著作権者およびTOPPERSプロジェクトを免責すること.
|
---|
25 | * また,本ソフトウェアのユーザまたはエンドユーザからのいかなる理
|
---|
26 | * 由に基づく請求からも,上記著作権者およびTOPPERSプロジェクトを
|
---|
27 | * 免責すること.
|
---|
28 | *
|
---|
29 | * 本ソフトウェアは,無保証で提供されているものである.上記著作権者お
|
---|
30 | * よびTOPPERSプロジェクトは,本ソフトウェアに関して,特定の使用目的
|
---|
31 | * に対する適合性も含めて,いかなる保証も行わない.また,本ソフトウェ
|
---|
32 | * アの利用により直接的または間接的に生じたいかなる損害に関しても,そ
|
---|
33 | * の責任を負わない.
|
---|
34 | *
|
---|
35 | * @(#) $Id: echonet.c 167 2016-03-08 11:37:45Z coas-nagasima $
|
---|
36 | */
|
---|
37 |
|
---|
38 | #include <kernel.h>
|
---|
39 | #include <stdio.h>
|
---|
40 | #include <string.h>
|
---|
41 | #include <t_stdlib.h>
|
---|
42 |
|
---|
43 | #include "echonet.h"
|
---|
44 | #include "echonet_fbs.h"
|
---|
45 | #include "echonet_task.h"
|
---|
46 | #include "echonet_dbg.h"
|
---|
47 | #include "echonet_agent.h"
|
---|
48 |
|
---|
49 | /*
|
---|
50 | * ECHONET Liteサービス処理開始
|
---|
51 | */
|
---|
52 | ER ecn_sta_svc()
|
---|
53 | {
|
---|
54 | return act_tsk(ecn_udp_taskid);
|
---|
55 | }
|
---|
56 |
|
---|
57 | /*
|
---|
58 | * ECHONETオブジェクト参照
|
---|
59 | * 引数
|
---|
60 | * ID fa_eobjid ECHONETオブジェクトID
|
---|
61 | * T_REOBJ *fp_eobj ECHONETオブジェクトの設定内容コピー先
|
---|
62 | *
|
---|
63 | * eobjidで指定したECHONETオブジェクトの設定内容を参照する。
|
---|
64 | * 参照した設定内容はpk_eobjに指定したメモリに返される。
|
---|
65 | */
|
---|
66 | ER ecn_ref_eobj(ID fa_eobjid, T_REOBJ *fp_eobj)
|
---|
67 | {
|
---|
68 | const EOBJINIB *p_obj;
|
---|
69 |
|
---|
70 | if (!fp_eobj)
|
---|
71 | return E_PAR; /* パラメータエラー */
|
---|
72 |
|
---|
73 | if ((fa_eobjid <= 0) || (fa_eobjid > tmax_eobjid))
|
---|
74 | return E_OBJ; /* オブジェクト未登録 */
|
---|
75 |
|
---|
76 | p_obj = &eobjinib_table[fa_eobjid - 1];
|
---|
77 |
|
---|
78 | *fp_eobj = *p_obj;
|
---|
79 |
|
---|
80 | return E_OK;
|
---|
81 | }
|
---|
82 |
|
---|
83 | /*
|
---|
84 | * ECHONETプロパティ参照
|
---|
85 | * 引数
|
---|
86 | * ID fa_eobjid ECHONETオブジェクトID
|
---|
87 | * uint8_t fa_epc プロパティコード
|
---|
88 | * T_RPRP *fp_eprp ECHONETオブジェクトのプロパティ設定内容コピー先
|
---|
89 | *
|
---|
90 | * eobjidで指定したECHONETオブジェクトの、epcで指定したプロパティの設定内容を参照する。
|
---|
91 | * 参照した設定内容はpk_eprpに指定したメモリに返される。
|
---|
92 | */
|
---|
93 | ER ecn_ref_eprp(ID fa_eobjid, uint8_t fa_epc, T_RPRP *fp_eprp)
|
---|
94 | {
|
---|
95 | const EOBJINIB *p_obj;
|
---|
96 | const EPRPINIB *p;
|
---|
97 | uint_t i;
|
---|
98 |
|
---|
99 | if (!fp_eprp)
|
---|
100 | return E_PAR; /* パラメータエラー */
|
---|
101 |
|
---|
102 | if ((fa_eobjid <= 0) || (fa_eobjid > tmax_eobjid))
|
---|
103 | return E_OBJ; /* オブジェクト未登録 */
|
---|
104 |
|
---|
105 | p_obj = &eobjinib_table[fa_eobjid - 1];
|
---|
106 |
|
---|
107 | /* eojの持つプロパティ初期化定義配列から検索 */
|
---|
108 | p = p_obj->eprp;
|
---|
109 | for (i = 0; i < p_obj->eprpcnt; i++, p++) {
|
---|
110 | if (p->eprpcd != fa_epc)
|
---|
111 | continue;
|
---|
112 | *fp_eprp = *p;
|
---|
113 | return E_OK;
|
---|
114 | }
|
---|
115 |
|
---|
116 | return E_PAR; /* パラメータエラー */
|
---|
117 | }
|
---|
118 |
|
---|
119 | /*
|
---|
120 | * プロパティ値書き込み要求(応答不要)電文作成
|
---|
121 | * 引数
|
---|
122 | * T_EDATA **ppk_esv 取得した電文の格納先
|
---|
123 | * ID fa_deoj 宛先のECHONETオブジェクトID
|
---|
124 | * uint8_t fa_epc プロパティコード
|
---|
125 | * uint8_t fa_pdc プロパティ値データサイズ
|
---|
126 | * const void *p_edt プロパティ値データ
|
---|
127 | *
|
---|
128 | * プロパティ値書き込み要求(応答不要)電文を作成する。
|
---|
129 | * 電文はdeojで指定したECHONETオブジェクトを宛先とし、
|
---|
130 | * 電文の初めのプロパティ値として、epcとpdc、p_edtで指定された
|
---|
131 | * プロパティコード、データサイズ、データを電文に設定する。
|
---|
132 | * 作成した電文の先頭アドレスはポインタ経由で返される。
|
---|
133 | */
|
---|
134 | ER ecn_esv_seti(T_EDATA **ppk_esv, ID fa_deoj, uint8_t fa_epc, uint8_t fa_pdc,
|
---|
135 | const void *p_edt)
|
---|
136 | {
|
---|
137 | return _ecn_tsk_mk_esv((ECN_FBS_ID *)ppk_esv, (ID)1, fa_deoj, fa_epc, fa_pdc, p_edt, ESV_SET_I);
|
---|
138 | }
|
---|
139 |
|
---|
140 | /*
|
---|
141 | * プロパティ値書き込み要求(応答要)電文作成
|
---|
142 | * 引数
|
---|
143 | * T_EDATA **ppk_esv 取得した電文の格納先
|
---|
144 | * ID fa_deoj 宛先のECHONETオブジェクトID
|
---|
145 | * uint8_t fa_epc プロパティコード
|
---|
146 | * uint8_t fa_pdc プロパティ値データサイズ
|
---|
147 | * const void *p_edt プロパティ値データ
|
---|
148 | *
|
---|
149 | * プロパティ値書き込み要求(応答要)電文を作成する。
|
---|
150 | * 電文はdeojで指定したECHONETオブジェクトを宛先とし、
|
---|
151 | * 電文の初めのプロパティ値として、epcとpdc、p_edtで指定された
|
---|
152 | * プロパティコード、データサイズ、データを電文に設定する。
|
---|
153 | * 作成した電文の先頭アドレスはポインタ経由で返される。
|
---|
154 | */
|
---|
155 | ER ecn_esv_setc(T_EDATA **ppk_esv, ID fa_deoj, uint8_t fa_epc, uint8_t fa_pdc,
|
---|
156 | const void *p_edt)
|
---|
157 | {
|
---|
158 | return _ecn_tsk_mk_esv((ECN_FBS_ID *)ppk_esv, (ID)1, fa_deoj, fa_epc, fa_pdc, p_edt, ESV_SET_C);
|
---|
159 | }
|
---|
160 |
|
---|
161 | /*
|
---|
162 | * プロパティ値読み出し要求電文作成
|
---|
163 | * 引数
|
---|
164 | * T_EDATA **ppk_esv 取得した電文の格納先
|
---|
165 | * ID fa_deoj 宛先のECHONETオブジェクトID
|
---|
166 | * uint8_t fa_epc プロパティコード
|
---|
167 | *
|
---|
168 | * プロパティ値読み出し要求電文を作成する。
|
---|
169 | * 電文はdeojで指定したECHONETオブジェクトを宛先とし、
|
---|
170 | * 電文の初めのプロパティ値として、epcで指定された
|
---|
171 | * プロパティコードを電文に設定する。
|
---|
172 | * 作成した電文の先頭アドレスはポインタ経由で返される。
|
---|
173 | */
|
---|
174 | ER ecn_esv_get(T_EDATA **ppk_esv, ID fa_deoj, uint8_t fa_epc)
|
---|
175 | {
|
---|
176 | return _ecn_tsk_mk_esv((ECN_FBS_ID *)ppk_esv, (ID)1, fa_deoj, fa_epc, 0, NULL, ESV_GET);
|
---|
177 | }
|
---|
178 |
|
---|
179 | /*
|
---|
180 | * プロパティ値通知要求電文作成
|
---|
181 | * 引数
|
---|
182 | * T_EDATA **ppk_esv 取得した電文の格納先
|
---|
183 | * ID fa_deoj 宛先のECHONETオブジェクトID
|
---|
184 | * uint8_t fa_epc プロパティコード
|
---|
185 | *
|
---|
186 | * プロパティ値通知要求電文を作成する。
|
---|
187 | * 電文はdeojで指定したECHONETオブジェクトを宛先とし、
|
---|
188 | * 電文の初めのプロパティ値として、epcで指定された
|
---|
189 | * プロパティコードを電文に設定する。
|
---|
190 | * 作成した電文の先頭アドレスはポインタ経由で返される。
|
---|
191 | */
|
---|
192 | ER ecn_esv_inf_req(T_EDATA **ppk_esv, ID fa_deoj, uint8_t fa_epc)
|
---|
193 | {
|
---|
194 | return _ecn_tsk_mk_esv((ECN_FBS_ID *)ppk_esv, (ID)1, fa_deoj, fa_epc, 0, NULL, ESV_INF_REQ);
|
---|
195 | }
|
---|
196 |
|
---|
197 | /*
|
---|
198 | * プロパティ値書き込み・読み出し要求電文作成
|
---|
199 | * 引数
|
---|
200 | * T_EDATA **ppk_esv 取得した電文の格納先
|
---|
201 | * ID fa_deoj 宛先のECHONETオブジェクトID
|
---|
202 | * uint8_t fa_epc プロパティコード
|
---|
203 | * uint8_t fa_pdc プロパティ値データサイズ
|
---|
204 | * const void *p_edt プロパティ値データ
|
---|
205 | *
|
---|
206 | * プロパティ値書き込み・読み出し要求電文を作成する。
|
---|
207 | * 電文はdeojで指定したECHONETオブジェクトを宛先とし、
|
---|
208 | * 電文の初めのプロパティ値として、epcとpdc、p_edtで指定された
|
---|
209 | * プロパティコード、データサイズ、データを電文に設定する。
|
---|
210 | * 作成した電文の先頭アドレスはポインタ経由で返される。
|
---|
211 | */
|
---|
212 | ER ecn_esv_set_get(T_EDATA **ppk_esv, ID fa_deoj, uint8_t fa_epc, uint8_t fa_pdc,
|
---|
213 | const void *p_edt)
|
---|
214 | {
|
---|
215 | return _ecn_tsk_mk_esv((ECN_FBS_ID *)ppk_esv, (ID)1, fa_deoj, fa_epc, fa_pdc, p_edt, ESV_SET_GET);
|
---|
216 | }
|
---|
217 |
|
---|
218 | /*
|
---|
219 | * プロパティ値書き込み・読み出し要求電文作成:折り返し指定
|
---|
220 | * プロパティ値書き込み・読み出し要求電文作成中の、書き込みプロパティ配列を終えて
|
---|
221 | * 読み出しプロパティ配列に移る時に用いる。
|
---|
222 | */
|
---|
223 | ER ecn_trn_set_get(T_EDATA *ppk_esv, int *p_trn_pos)
|
---|
224 | {
|
---|
225 | ER a_ret;
|
---|
226 |
|
---|
227 | if (!ppk_esv)
|
---|
228 | return E_PAR;
|
---|
229 | if (!p_trn_pos)
|
---|
230 | return E_PAR;
|
---|
231 |
|
---|
232 | /* プロパティ読み込み件数書き込み時のヘッド情報を記録 */
|
---|
233 | *p_trn_pos = _ecn_fbs_get_datalen(*(ECN_FBS_ID *)&ppk_esv);
|
---|
234 |
|
---|
235 | /* 応答電文用メモリにデータ追加・この時点での応答電文中プロパティ件数を記録 */
|
---|
236 | a_ret = _ecn_fbs_add_data_ex(*(ECN_FBS_ID *)&ppk_esv, &ppk_esv->hdr.edata.opc, 1);
|
---|
237 | if (a_ret) {
|
---|
238 | ECN_DBG_PUT( "ecn_trn_set_get(): _ecn_fbs_add_data_ex() result = %d:%s", a_ret, itron_strerror(a_ret) );
|
---|
239 | return a_ret;
|
---|
240 | }
|
---|
241 |
|
---|
242 | /* 応答電文中プロパティ件数を 0 に戻す(今後、終了指定するまでは読み出しプロパティ件数になる) */
|
---|
243 | ppk_esv->hdr.edata.opc = 0;
|
---|
244 |
|
---|
245 | return E_OK;
|
---|
246 | }
|
---|
247 |
|
---|
248 | /*
|
---|
249 | * プロパティ値書き込み・読み出し要求電文作成:終了指定
|
---|
250 | * プロパティ値書き込み・読み出し要求電文作成中の、読み出しプロパティ配列を終える時に用いる。
|
---|
251 | */
|
---|
252 | ER ecn_end_set_get(T_EDATA *ppk_esv, int fa_trn_pos)
|
---|
253 | {
|
---|
254 | ER a_ret;
|
---|
255 | int a_wr_opc;
|
---|
256 | uint8_t a_rd_opc;
|
---|
257 |
|
---|
258 | if (!ppk_esv)
|
---|
259 | return E_PAR;
|
---|
260 |
|
---|
261 | /* この時点での応答電文中プロパティ件数(読み出しプロパティ件数)を保持 */
|
---|
262 | a_rd_opc = ppk_esv->hdr.edata.opc;
|
---|
263 |
|
---|
264 | /* 応答電文中プロパティ件数(書き込みプロパティ件数)を保存位置から読み取り */
|
---|
265 | a_wr_opc = _ecn_fbs_peek(*(ECN_FBS_ID *)&ppk_esv, fa_trn_pos);
|
---|
266 | if (a_wr_opc < 0) {
|
---|
267 | ECN_DBG_PUT( "ecn_end_set_get(): _ecn_fbs_peek() result = %d:%s", a_wr_opc, itron_strerror(a_wr_opc) );
|
---|
268 | return a_wr_opc;
|
---|
269 | }
|
---|
270 |
|
---|
271 | /* 応答電文中プロパティ件数(書き込みプロパティ件数)を復元 */
|
---|
272 | ppk_esv->hdr.edata.opc = (uint8_t)a_wr_opc;
|
---|
273 |
|
---|
274 | /* 応答電文中プロパティ件数(読み出しプロパティ件数)を保存位置に書き込み */
|
---|
275 | a_ret = _ecn_fbs_poke(*(ECN_FBS_ID *)&ppk_esv, fa_trn_pos, a_rd_opc);
|
---|
276 | if (a_ret) {
|
---|
277 | ECN_DBG_PUT( "ecn_end_set_get(): _ecn_fbs_poke() result = %d:%s", a_ret, itron_strerror(a_ret) );
|
---|
278 | return a_ret;
|
---|
279 | }
|
---|
280 |
|
---|
281 | return E_OK;
|
---|
282 | }
|
---|
283 |
|
---|
284 | /*
|
---|
285 | * プロパティ値通知(応答要)電文作成
|
---|
286 | * 引数
|
---|
287 | * T_EDATA **ppk_esv 取得した電文の格納先
|
---|
288 | * ID fa_deoj 宛先のECHONETオブジェクトID
|
---|
289 | * ID fa_seoj 送信元のECHONETオブジェクトID
|
---|
290 | * uint8_t fa_sepc 送信元のプロパティコード
|
---|
291 | */
|
---|
292 | ER ecn_esv_infc(T_EDATA **ppk_esv, ID fa_deoj, ID fa_seoj, uint8_t fa_sepc)
|
---|
293 | {
|
---|
294 | return _ecn_tsk_mk_esv((ECN_FBS_ID *)ppk_esv, fa_seoj, fa_deoj, fa_sepc, 0, NULL, ESV_INFC);
|
---|
295 | }
|
---|
296 |
|
---|
297 | /*
|
---|
298 | * 要求電文へのプロパティ指定追加 (プロパティデータが付随しない場合に用いる)
|
---|
299 | */
|
---|
300 | ER ecn_add_epc(T_EDATA *pk_esv, uint8_t epc)
|
---|
301 | {
|
---|
302 | return ecn_add_edt(pk_esv, epc, 0, 0);
|
---|
303 | }
|
---|
304 |
|
---|
305 | /*
|
---|
306 | * 要求電文へのプロパティデータ追加 (プロパティおよび付随データを追加する)
|
---|
307 | */
|
---|
308 | ER ecn_add_edt(T_EDATA *pk_esv, uint8_t fa_epc, uint8_t fa_pdc, const void *p_edt)
|
---|
309 | {
|
---|
310 | ECN_FBS_ID a_fbs_id;
|
---|
311 | ER a_ret;
|
---|
312 | T_ECN_PRP a_ecn_prp;
|
---|
313 |
|
---|
314 | if (!pk_esv)
|
---|
315 | return E_PAR; /* 取得したFBS_IDの格納先がNULL */
|
---|
316 | if (ECHONET_MEMPOOL_BLOCK_SIZE <= fa_pdc)
|
---|
317 | return E_PAR; /* プロパティ値サイズが大きすぎる */
|
---|
318 | if (!p_edt && 0 < fa_pdc)
|
---|
319 | return E_PAR; /* プロパティ値サイズが1以上なのにデータポインタがNULL */
|
---|
320 |
|
---|
321 | a_fbs_id.ptr = (T_ECN_FST_BLK *)pk_esv;
|
---|
322 | if (((T_EDATA *)a_fbs_id.ptr)->hdr.edata.opc == 0xFF)
|
---|
323 | return E_PAR; /* プロパティが多すぎる */
|
---|
324 |
|
---|
325 | /* 要求電文用メモリにデータ追加 */
|
---|
326 | memset(&a_ecn_prp, 0, sizeof(a_ecn_prp));
|
---|
327 | a_ecn_prp.epc = fa_epc;
|
---|
328 | a_ecn_prp.pdc = fa_pdc;
|
---|
329 | a_ret = _ecn_fbs_add_data_ex(a_fbs_id, &a_ecn_prp, sizeof(a_ecn_prp));
|
---|
330 | if (a_ret) {
|
---|
331 | ECN_DBG_PUT( "ecn_add_edt(): _ecn_fbs_add_data_ex(*, ecn_prp{epc:0x%02X}, %u) result = %d:%s",
|
---|
332 | a_ecn_prp.epc, sizeof(a_ecn_prp),
|
---|
333 | a_ret, itron_strerror(a_ret) );
|
---|
334 | goto lb_except;
|
---|
335 | }
|
---|
336 | if (0 < fa_pdc) {
|
---|
337 | /* 付随データ追加 */
|
---|
338 | a_ret = _ecn_fbs_add_data_ex(a_fbs_id, p_edt, fa_pdc);
|
---|
339 | if (a_ret) {
|
---|
340 | ECN_DBG_PUT( "ecn_add_edt(): _ecn_fbs_add_data_ex(*, ecn_prp{epc:0x%02X} edt, %u) result = %d:%s",
|
---|
341 | a_ecn_prp.epc, fa_pdc,
|
---|
342 | a_ret, itron_strerror(a_ret) );
|
---|
343 | goto lb_except;
|
---|
344 | }
|
---|
345 | }
|
---|
346 | /* プロパティ数インクリメント */
|
---|
347 | ((T_EDATA *)a_fbs_id.ptr)->hdr.edata.opc++;
|
---|
348 |
|
---|
349 | lb_except:
|
---|
350 | return a_ret;
|
---|
351 | }
|
---|
352 |
|
---|
353 | /*
|
---|
354 | * 要求電文の送信
|
---|
355 | * esvで指定された要求電文を送信する。
|
---|
356 | * 電文に指定された宛先からIPアドレスを特定し、UDPパケットとして送信する。
|
---|
357 | */
|
---|
358 | ER ecn_snd_esv(T_EDATA *pk_esv)
|
---|
359 | {
|
---|
360 | ECN_FBS_ID a_rsp_fbs;
|
---|
361 | a_rsp_fbs.ptr = (T_ECN_FST_BLK *)pk_esv;
|
---|
362 | return _ecn_tsk_snd_mbx(a_rsp_fbs, true);
|
---|
363 | }
|
---|
364 |
|
---|
365 | /*
|
---|
366 | * 応答電文の受信永遠待ち
|
---|
367 | * 応答電文を受信するのを待つ。
|
---|
368 | * 受信した応答電文はppk_esvで指定したポインターに先頭アドレスが返される。
|
---|
369 | * 要求電文の宛先のECHONETオブジェクトがネットワークに存在しない場合、
|
---|
370 | * 応答電文は返送されないので、永遠に待つことになるのでタイムアウト付きの
|
---|
371 | * ecn_trcv_esvを使用することを推奨する。
|
---|
372 | */
|
---|
373 | ER ecn_rcv_esv(T_EDATA **ppk_esv)
|
---|
374 | {
|
---|
375 | return ecn_trcv_esv(ppk_esv, TMO_FEVR);
|
---|
376 | }
|
---|
377 |
|
---|
378 | /*
|
---|
379 | * 応答電文の受信待ちタイムアウトあり
|
---|
380 | * 応答電文を受信するのを待つ。
|
---|
381 | * 受信した応答電文はppk_esvで指定したポインターに先頭アドレスが返される。
|
---|
382 | * tmoで指定されたタイムアウト時間が経過しても応答電文を受信しない場合、
|
---|
383 | * 待ちが解除され、戻り値がE_TMOで返される。
|
---|
384 | * tmoにTMO_FEVRを指定するとecn_rcv_esvと同じ振る舞いをする。
|
---|
385 | * tmoに0を指定するとecn_prcv_esvと同じ振る舞いをする。
|
---|
386 | */
|
---|
387 | ER ecn_trcv_esv(T_EDATA **ppk_esv, TMO fa_tmout)
|
---|
388 | {
|
---|
389 | T_MSG *p_msg = 0;
|
---|
390 | ER a_ret;
|
---|
391 |
|
---|
392 | if (!ppk_esv)
|
---|
393 | return E_PAR;
|
---|
394 |
|
---|
395 | a_ret = trcv_mbx(ecn_api_mailboxid, &p_msg, fa_tmout);
|
---|
396 | if (a_ret != E_OK) {
|
---|
397 | *ppk_esv = NULL;
|
---|
398 | return a_ret;
|
---|
399 | }
|
---|
400 |
|
---|
401 | *ppk_esv = (T_EDATA *)p_msg;
|
---|
402 | return (((T_ECN_FST_BLK *)p_msg)->hdr.type == ECN_MSG_ECHONET) ? E_OK : E_WBLK;
|
---|
403 | }
|
---|
404 |
|
---|
405 | /*
|
---|
406 | * 応答電文の受信ポーリング
|
---|
407 | * 応答電文を受信するのを待つ。
|
---|
408 | * 受信した応答電文はppk_esvで指定したポインターに先頭アドレスが返される。
|
---|
409 | * 応答電文の受信の有無にかかわらず、待たずに関数を抜ける。
|
---|
410 | */
|
---|
411 | ER ecn_prcv_esv(T_EDATA **ppk_esv)
|
---|
412 | {
|
---|
413 | return ecn_trcv_esv(ppk_esv, TMO_POL);
|
---|
414 | }
|
---|
415 |
|
---|
416 | /*
|
---|
417 | * 応答電文の破棄
|
---|
418 | */
|
---|
419 | ER ecn_rel_esv(T_EDATA *pk_esv)
|
---|
420 | {
|
---|
421 | ECN_FBS_ID a_fbs;
|
---|
422 | ER a_ret;
|
---|
423 |
|
---|
424 | a_fbs.ptr = (T_ECN_FST_BLK *)pk_esv;
|
---|
425 | a_ret = _ecn_fbs_del(a_fbs);
|
---|
426 | return a_ret;
|
---|
427 | }
|
---|
428 |
|
---|
429 | /*
|
---|
430 | * 応答電文の送信元機器オブジェクトを取得する
|
---|
431 | */
|
---|
432 | ID ecn_get_eobj(T_EDATA *pk_esv)
|
---|
433 | {
|
---|
434 | const EOBJCB *p_snod;
|
---|
435 | const EOBJINIB *p_sobj = NULL;
|
---|
436 | T_ECN_EDT_HDR *p_esv;
|
---|
437 |
|
---|
438 | p_esv = &pk_esv->hdr;
|
---|
439 | p_snod = _ecn_eno_fnd(((T_ECN_FST_BLK *)pk_esv)->hdr.sender.id);
|
---|
440 | if (p_snod != NULL) {
|
---|
441 | /* ノードプロファイルの場合 */
|
---|
442 | if ((p_esv->edata.seoj.eojx1 == EOJ_X1_PROFILE)
|
---|
443 | && (p_esv->edata.seoj.eojx2 == EOJ_X2_NODE_PROFILE)) {
|
---|
444 | p_sobj = p_snod->profile;
|
---|
445 | }
|
---|
446 | /* 機器オブジェクトの場合 */
|
---|
447 | else {
|
---|
448 | p_sobj = _ecn_eoj_fnd(p_snod, &p_esv->edata.seoj);
|
---|
449 | }
|
---|
450 | }
|
---|
451 |
|
---|
452 | if (p_sobj == NULL)
|
---|
453 | return EOBJ_NULL;
|
---|
454 | #ifdef ECHONET_CONTROLLER_EXTENTION
|
---|
455 | if (p_sobj->eprpcnt == 0)
|
---|
456 | return ecn_agent_get_eobj(p_sobj);
|
---|
457 | #endif
|
---|
458 | return 1 + (((intptr_t)p_sobj - (intptr_t)eobjinib_table) / sizeof(EOBJINIB));
|
---|
459 | }
|
---|
460 |
|
---|
461 | /*
|
---|
462 | * 応答電文解析イテレーター初期化
|
---|
463 | */
|
---|
464 | ER ecn_itr_ini(T_ENUM_EPC *pk_itr, T_EDATA *pk_esv)
|
---|
465 | {
|
---|
466 | if (!pk_itr)
|
---|
467 | return E_PAR;
|
---|
468 | if (!pk_esv)
|
---|
469 | return E_PAR;
|
---|
470 |
|
---|
471 | memset(pk_itr, 0, sizeof(*pk_itr));
|
---|
472 | pk_itr->pk_esv = pk_esv;
|
---|
473 | pk_itr->count = pk_esv->hdr.edata.opc;
|
---|
474 | /* 今のブロックでのプロパティ総数 */
|
---|
475 | pk_itr->got_ct = 0; /* 今のブロックで、読み取った数 */
|
---|
476 | pk_itr->next_blk_ct = 0; /* 後続ブロック数 */
|
---|
477 | pk_itr->is_eof = 0; /* 終端に達した時、非0 */
|
---|
478 | pk_itr->cur = offsetof(T_ECN_EDT_HDR, ecn_prp);
|
---|
479 |
|
---|
480 | switch (pk_esv->hdr.edata.esv) {
|
---|
481 | case ESV_SET_GET: /* 0x6E プロパティ値書き込み・読み出し要求 */
|
---|
482 | case ESV_SET_GET_RES: /* 0x7E プロパティ値書き込み・読み出し応答 */
|
---|
483 | case ESV_SET_GET_SNA: /* 0x5E プロパティ値書き込み・読み出し不可応答 */
|
---|
484 | pk_itr->next_blk_ct ++;
|
---|
485 | default:
|
---|
486 | break;
|
---|
487 | }
|
---|
488 |
|
---|
489 | return E_OK;
|
---|
490 | }
|
---|
491 |
|
---|
492 | /*
|
---|
493 | * 応答電文解析イテレーターインクリメント
|
---|
494 | */
|
---|
495 | ER ecn_itr_nxt(T_ENUM_EPC *pk_itr, uint8_t *p_epc, uint8_t *p_pdc, void *p_edt)
|
---|
496 | {
|
---|
497 | ECN_FBS_ID a_fbs_id;
|
---|
498 | int a_rd_bak;
|
---|
499 | ER a_ret;
|
---|
500 | ECN_FBS_SSIZE_T a_rd_len;
|
---|
501 | T_ECN_PRP a_ecn_prp;
|
---|
502 |
|
---|
503 | if (!pk_itr)
|
---|
504 | return E_PAR;
|
---|
505 | if (!p_epc)
|
---|
506 | return E_PAR;
|
---|
507 | if (!p_pdc)
|
---|
508 | return E_PAR;
|
---|
509 | if (!p_edt)
|
---|
510 | return E_PAR;
|
---|
511 | if (!pk_itr->pk_esv)
|
---|
512 | return E_PAR;
|
---|
513 |
|
---|
514 | if (pk_itr->is_eof)
|
---|
515 | return E_BOVR; /* データ終了 */
|
---|
516 | if ( pk_itr->count <= pk_itr->got_ct
|
---|
517 | && pk_itr->next_blk_ct < 1) {
|
---|
518 | pk_itr->is_eof = 1; /* 終端に達した時、非0 */
|
---|
519 | return E_BOVR; /* データ終了 */
|
---|
520 | }
|
---|
521 |
|
---|
522 | a_fbs_id.ptr = (T_ECN_FST_BLK *)pk_itr->pk_esv;
|
---|
523 |
|
---|
524 | /* 元のカーソル位置を保存し、イテレータのカーソル位置にする */
|
---|
525 | a_rd_bak = a_fbs_id.ptr->hdr.rd;
|
---|
526 | a_fbs_id.ptr->hdr.rd = pk_itr->cur;
|
---|
527 |
|
---|
528 | #ifdef ECN_ENA_ITR_NXT_CARP
|
---|
529 | ECN_CAP_PUT( "ecn_itr_nxt() rd.cur=b%dp%d", pk_itr->cur.blk, pk_itr->cur.pos ECN_CAP_PUT(;
|
---|
530 | #endif
|
---|
531 | if (pk_itr->count <= pk_itr->got_ct && 0 < pk_itr->next_blk_ct) {
|
---|
532 | /* 次ブロックに移動 */
|
---|
533 | pk_itr->next_blk_ct --;
|
---|
534 | pk_itr->count = 0; /* 今のブロックでのプロパティ総数 */
|
---|
535 | pk_itr->got_ct = 0; /* 今のブロックで、読み取った数 */
|
---|
536 |
|
---|
537 | /* 次ブロックのプロパティ数を読み取る */
|
---|
538 | a_rd_len = 0;
|
---|
539 | a_ret = _ecn_fbs_get_data(a_fbs_id, &a_ecn_prp.pdc, 1, &a_rd_len);
|
---|
540 | if (a_ret != E_OK)
|
---|
541 | goto lb_except;
|
---|
542 | if (0 < a_rd_len) {
|
---|
543 | pk_itr->count = a_ecn_prp.pdc; /* 今のブロックでのプロパティ総数 */
|
---|
544 | }
|
---|
545 | a_ret = E_BOVR; /* データ終了 */
|
---|
546 | goto lb_finally;
|
---|
547 | }
|
---|
548 |
|
---|
549 | /* プロパティコードとデータサイズを読み取る */
|
---|
550 | memset(&a_ecn_prp, 0, sizeof(a_ecn_prp));
|
---|
551 | a_rd_len = 0;
|
---|
552 | a_ret = _ecn_fbs_get_data(a_fbs_id, &a_ecn_prp, sizeof(a_ecn_prp), &a_rd_len);
|
---|
553 | if (a_ret != E_OK)
|
---|
554 | goto lb_except;
|
---|
555 | if (a_rd_len < sizeof(a_ecn_prp)) {
|
---|
556 | ECN_DBG_PUT( "ecn_itr_nxt() ecn_prp read fault."
|
---|
557 | " rd.cur=%d", pk_itr->cur );
|
---|
558 | pk_itr->is_eof = 1; /* 終端に達した時、非0 */
|
---|
559 | a_ret = E_BOVR; /* データ終了 */
|
---|
560 | goto lb_finally;
|
---|
561 | }
|
---|
562 | *p_epc = a_ecn_prp.epc;
|
---|
563 | *p_pdc = a_ecn_prp.pdc;
|
---|
564 |
|
---|
565 | if (0 < a_ecn_prp.pdc) {
|
---|
566 | /* 付随データを読み取る */
|
---|
567 | a_rd_len = 0;
|
---|
568 | a_ret = _ecn_fbs_get_data(a_fbs_id, p_edt, a_ecn_prp.pdc, &a_rd_len);
|
---|
569 | if (a_ret != E_OK)
|
---|
570 | goto lb_except;
|
---|
571 | if (a_rd_len < (ECN_FBS_SSIZE_T)a_ecn_prp.pdc) {
|
---|
572 | ECN_DBG_PUT( "ecn_itr_nxt() edt read fault."
|
---|
573 | " rd.cur=%d,epc=0x%02X,pdc=%u", pk_itr->cur, a_ecn_prp.epc , a_ecn_prp.pdc );
|
---|
574 | pk_itr->is_eof = 1; /* 終端に達した時、非0 */
|
---|
575 | a_ret = E_BOVR; /* データ終了 */
|
---|
576 | goto lb_finally;
|
---|
577 | }
|
---|
578 | }
|
---|
579 | pk_itr->got_ct++;
|
---|
580 |
|
---|
581 | #ifdef ECN_ENA_ITR_NXT_CARP
|
---|
582 | ECN_CAP_PUT( "ecn_itr_nxt() read: ct=%d/%d", pk_itr->got_ct, pk_itr->count ECN_CAP_PUT(;
|
---|
583 | #endif
|
---|
584 |
|
---|
585 | lb_finally:
|
---|
586 | /* イテレータのカーソル位置を更新 */
|
---|
587 | pk_itr->cur = a_fbs_id.ptr->hdr.rd;
|
---|
588 |
|
---|
589 | lb_except:
|
---|
590 | /* 元のカーソル位置に戻す */
|
---|
591 | a_fbs_id.ptr->hdr.rd = a_rd_bak;
|
---|
592 | return a_ret;
|
---|
593 | }
|
---|
594 |
|
---|
595 | /*
|
---|
596 | * 応答電文待ちの割り込み送信
|
---|
597 | */
|
---|
598 | ER ecn_brk_wai(const void *p_dat, int fa_datsz)
|
---|
599 | {
|
---|
600 | ER a_ret;
|
---|
601 | ECN_FBS_ID a_fbs_id = { 0 }; /* 要求電文用メモリ */
|
---|
602 |
|
---|
603 | /* 応答電文待ちの割り込み電文作成 */
|
---|
604 | a_ret = _ecn_mk_brk_wai(&a_fbs_id, p_dat, fa_datsz);
|
---|
605 | if (a_ret)
|
---|
606 | return a_ret;
|
---|
607 |
|
---|
608 | /* 割り込み送信 */
|
---|
609 | a_ret = _ecn_tsk_snd_mbx(a_fbs_id, true);
|
---|
610 | if (a_ret) {
|
---|
611 | ECN_DBG_PUT( "_ecn_tsk_snd_mbx() result = %d:%s",
|
---|
612 | a_ret, itron_strerror(a_ret) );
|
---|
613 | goto lb_except;
|
---|
614 | }
|
---|
615 |
|
---|
616 | /* 正常終了 */
|
---|
617 | return a_ret;
|
---|
618 |
|
---|
619 | lb_except:
|
---|
620 | /* データ作成失敗したら領域解放 */
|
---|
621 | if (a_fbs_id.ptr)
|
---|
622 | _ecn_fbs_del(a_fbs_id);
|
---|
623 |
|
---|
624 | return a_ret;
|
---|
625 | }
|
---|
626 |
|
---|
627 | /*
|
---|
628 | * 割り込みデータの取得
|
---|
629 | */
|
---|
630 | ER ecn_get_brk_dat(T_EDATA *pk_esv, void *p_buf, int fa_bufsz, int *p_datsz)
|
---|
631 | {
|
---|
632 | ECN_FBS_ID a_fbs;
|
---|
633 | ER a_ret;
|
---|
634 | ECN_FBS_SSIZE_T a_len;
|
---|
635 |
|
---|
636 | if (p_datsz)
|
---|
637 | *p_datsz = 0;
|
---|
638 | if (!pk_esv)
|
---|
639 | return E_PAR;
|
---|
640 |
|
---|
641 | a_fbs.ptr = (T_ECN_FST_BLK *)pk_esv;
|
---|
642 | if (a_fbs.ptr->hdr.type != ECN_MSG_USER_BREAK) {
|
---|
643 | ECN_DBG_PUT( "ecn_get_brk_dat: fbs-type:%d != ECN_MSG_USER_BREAK",
|
---|
644 | a_fbs.ptr->hdr.type );
|
---|
645 | return E_PAR;
|
---|
646 | }
|
---|
647 |
|
---|
648 | /* 読み取り位置を先頭に戻す */
|
---|
649 | a_ret = _ecn_fbs_set_rpos(a_fbs, 0);
|
---|
650 | if (a_ret) {
|
---|
651 | ECN_DBG_PUT( "_ecn_fbs_set_rpos(*, 0) result = %d:%s",
|
---|
652 | a_ret, itron_strerror(a_ret) );
|
---|
653 | return a_ret;
|
---|
654 | }
|
---|
655 |
|
---|
656 | /* 付随データを読み込む */
|
---|
657 | if (p_buf && 0 < fa_bufsz) {
|
---|
658 | a_len = 0;
|
---|
659 | a_ret = _ecn_fbs_get_data(a_fbs, p_buf, fa_bufsz, &a_len);
|
---|
660 | if (a_ret || !a_len) {
|
---|
661 | ECN_DBG_PUT( "_ecn_fbs_get_data(*, p_buf, fa_bufsz) result = %d:%s",
|
---|
662 | a_ret, itron_strerror(a_ret) );
|
---|
663 | return E_PAR;
|
---|
664 | }
|
---|
665 | *p_datsz = a_len;
|
---|
666 | }
|
---|
667 |
|
---|
668 | return E_OK;
|
---|
669 | }
|
---|
670 |
|
---|
671 | #ifndef ECN_USER_DATA_PROP_SET
|
---|
672 | /*
|
---|
673 | * データ設定関数
|
---|
674 | */
|
---|
675 | int ecn_data_prop_set(const EPRPINIB *item, const void *src, int size, bool_t *anno)
|
---|
676 | {
|
---|
677 | if (size > item->eprpsz)
|
---|
678 | size = item->eprpsz;
|
---|
679 |
|
---|
680 | if (*anno)
|
---|
681 | *anno = memcmp((uint8_t *)item->exinf, src, size) != 0;
|
---|
682 |
|
---|
683 | memcpy((uint8_t *)item->exinf, src, size);
|
---|
684 |
|
---|
685 | return size;
|
---|
686 | }
|
---|
687 | #endif
|
---|
688 |
|
---|
689 | #ifndef ECN_USER_DATA_PROP_GET
|
---|
690 | /*
|
---|
691 | * データ取得関数
|
---|
692 | */
|
---|
693 | int ecn_data_prop_get(const EPRPINIB *item, void *dst, int size)
|
---|
694 | {
|
---|
695 | if (size > item->eprpsz)
|
---|
696 | size = item->eprpsz;
|
---|
697 |
|
---|
698 | memcpy(dst, (uint8_t *)item->exinf, size);
|
---|
699 |
|
---|
700 | return size;
|
---|
701 | }
|
---|
702 | #endif
|
---|