ipc_call.c
16.6 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
/**
@file ipc_call.c
@date 2008-06-30
@author 박진호 jhpark@falinux.com
@brief 프로세스간의 명령 전달을 구현한다.
@modify
@todo
@bug
@remark
@warning
*/
//
// 저작권 에프에이리눅스(주)
// 외부공개 금지
//
//----------------------------------------------------------------------------
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/signal.h>
#include <sys/ioctl.h>
#include <sys/poll.h>
#include <time.h>
#include <errno.h>
#include <pollmng.h>
#include <fabind.h>
#include <uds.h>
#include <udp.h>
#include <ipc_call.h>
/// @brief 전역 변수 선언
//------------------------------------------------------------------------------
/// @}
char ipc_call_desc[] = "ipc_call ver 0.3.0";
#define FABIND_QUEUE_COUNT 1024
/// fabind 개별 구조체
typedef struct {
unsigned long id; ///
unsigned long port; /// 통신을 위한 PORT
unsigned long group; /// 구룹 처리를 위한 GROUP ID
long method_type; /// 통신 처리 방법 ( FABIND, UDS, UDP )
unsigned long level; /// LEVEL 값
unsigned long time_out; /// TIME OUT 값
poll_obj_t *poll_obj; /// 폴 관련 obj
ipc_call_func_t *func; /// 데이타 수신시 호출하는 사용자 콜백함수
ipc_call_packet_t recv_packet; /// 데이타 수신 패켓
} ipc_priv_t;
/// IPC_CALL 의 변수를 관리하는 구조체
static ipc_priv_t ipc_priv;
int ipc_call_poll_in( poll_obj_t *obj);
int ipc_call_method ( unsigned long method );
//------------------------------------------------------------------------------
/** @brief IPC_CALL 에러 상태를 출력 한다.
@param err err 넘버
*///----------------------------------------------------------------------------
void ipc_call_print_error ( int err )
{
if ( 0 > err )
{
switch( err )
{
case IPC_CALL_E_AGAIN : printf("ipc_call) Error Again\n"); break;
case IPC_CALL_E_TIME_OUT : printf("ipc_call) Error Timeout\n"); break;
case IPC_CALL_E_BUSY : printf("ipc_call) Error Busy\n"); break;
case IPC_CALL_E_NO_ID : printf("ipc_call) Error No id\n"); break;
}
}
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 을 생성한다.
@param id IPC의 ID.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_create ( unsigned long id )
{
poll_obj_t *obj;
memset( (void *)&ipc_priv, 0, sizeof(ipc_priv_t) );
ipc_priv.id = id;
ipc_priv.port = IPC_CALL_GET_PORT(id) ;
ipc_priv.group = IPC_CALL_GET_GROUP(id);
ipc_priv.method_type = IPC_CALL_METHOD_NONE;
// 디폴트로 UDS로 열고 다른것은 타입을 변경할때 처리 하도록 한다.
return ipc_call_method( IPC_CALL_METHOD_UDS );
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 을 해제한다.
@param id IPC의 ID.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_free( unsigned long id )
{
poll_obj_t *obj;
if ( ipc_priv.id != id ) return IPC_CALL_E_NO_ID;
obj = ipc_priv.poll_obj;
switch( ipc_priv.method_type )
{
case IPC_CALL_METHOD_NONE : break;
case IPC_CALL_METHOD_FABIND : fabind_close( obj ); break;
case IPC_CALL_METHOD_UDS : uds_close( obj ); break;
case IPC_CALL_METHOD_UDP : udp_close( obj ); break;
default : printf("Not Match Method\n"); break;
}
return IPC_CALL_OK;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 수행 방식을 정한다.
@param method 메시지 전송 방법
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_method ( unsigned long method )
{
poll_obj_t *obj = NULL;
if ( ipc_priv.method_type == method ) return IPC_CALL_OK;
ipc_call_free( ipc_priv.id );
switch( method )
{
case IPC_CALL_METHOD_FABIND :
{
obj = fabind_open( ipc_priv.port, ipc_priv.group, FABIND_QUEUE_COUNT);
if ( NULL == obj )
{
return IPC_CALL_E_BUSY;
}
ipc_priv.method_type = IPC_CALL_METHOD_FABIND;
}
break;
case IPC_CALL_METHOD_UDS :
{
char dst_uds[1024];
sprintf( dst_uds, IPC_CALL_UDS_NAME_FORMAT, ipc_priv.port );
obj = uds_open_server( dst_uds );
if ( NULL == obj )
{
return IPC_CALL_E_BUSY;
}
ipc_priv.method_type = IPC_CALL_METHOD_UDS;
}
break;
case IPC_CALL_METHOD_UDP :
{
obj = udp_open_server( ipc_priv.port );
if ( NULL == obj )
{
return IPC_CALL_E_BUSY;
}
ipc_priv.method_type = IPC_CALL_METHOD_UDP;
}
break;
default : printf("Not Match METHOD\n"); break;
}
if( NULL != obj )
{
obj->on_poll_in = ipc_call_poll_in;
ipc_priv.poll_obj = obj;
}
return IPC_CALL_OK;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 레벨을 정한다.
@param level 전송 레벨을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_level ( unsigned long level )
{
ipc_priv.level = level;
return 0;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 디폴트 레벨을 설정한다.
@param level 디폴트 전송 레벨을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_level_default ( unsigned long level )
{
ipc_priv.level = level;
return 0;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 시간 초과를 설정한다.
@param msec timeout될 시간을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_timeout ( unsigned long msec )
{
ipc_priv.time_out = msec;
return 0;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 디폴트 시간 초과를 정한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_timeout_default ( unsigned long msec )
{
ipc_priv.time_out = msec;
return 0;
}
//------------------------------------------------------------------------------
/** @brief POLLIN을 위한 함수 포인터
@param obj poll_obj
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_poll_in( poll_obj_t *obj)
{
ipc_call_packet_t *packet;
int ack, rsize = 0;
packet = &(ipc_priv.recv_packet);
switch( ipc_priv.method_type )
{
case IPC_CALL_METHOD_FABIND : rsize = fabind_read( obj, (void *)packet, sizeof(ipc_call_packet_t), &ack ); break;
case IPC_CALL_METHOD_UDS : rsize = uds_read( obj, (void *)packet, sizeof(ipc_call_packet_t) ); break;
case IPC_CALL_METHOD_UDP : rsize = udp_read( obj, (void *)packet, sizeof(ipc_call_packet_t) ); break;
default : printf("Not Match Method\n"); break;
}
#if 0
{
int i;
printf("rsize:%d ======================\n", rsize);
for( i = 0; i < packet->size; i++)
{
if ( i%10 == 0 ) printf("\n");
printf("%X ", packet->buf[i]);
}
printf("\n=============================\n");
}
#endif
if ( ( 0 < rsize ) && ipc_priv.func )
{
ipc_call_func_t read_func;
read_func = ipc_priv.func;
read_func( packet->src_id, packet->msg_type, packet->buf, packet->size );
}
return POLL_EVENTED;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 읽기에 대한 이벤트를 지정한다.
@param func read시 호출될 함수 포인터를 설정 한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_on_read ( ipc_call_func_t *func )
{
ipc_priv.func = func;
return 0;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 수신된 버퍼의 내용을 BIN 형식으로 읽는다.
@param msec timeout될 디폴트 시간을 정한다.
@return 0 성공
-1 실패
@remark 이 함수는 필요가 없을것 같다. 판단필요
*///----------------------------------------------------------------------------
int ipc_call_read ( void *buf, int len )
{
poll_obj_t *obj;
int rsize = 0;
int ack;
char rbuf[1024];
memset( rbuf, 0x00, sizeof(rbuf) );
obj = ipc_priv.poll_obj;
switch( ipc_priv.method_type )
{
case IPC_CALL_METHOD_FABIND : rsize = fabind_read( obj, rbuf, len, &ack ); break;
case IPC_CALL_METHOD_UDS : rsize = uds_read( obj, rbuf, len ); break;
case IPC_CALL_METHOD_UDP : rsize = udp_read( obj, rbuf, len ); break;
default : printf("Not Match METHOD\n"); break;
}
memcpy( buf, rbuf, rsize );
return 0;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 BIN 형식으로 데이터를 전송한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_send ( unsigned long dest, void *buf, int len, unsigned long msg_type )
{
poll_obj_t *obj;
ipc_call_packet_t packet;
int ret = 0;
// printf( "IPC_CALL_SEND DATA[%s] LEN=%d\n", buf, len );
if ( 0 >= len ) return IPC_CALL_E_EMPTY;
if ( len > IPC_CALL_BUF_LEN ) len = IPC_CALL_BUF_LEN;
obj = ipc_priv.poll_obj;
packet.dst_id = dest;
packet.src_id = ipc_priv.id;
packet.msg_type = msg_type;
packet.size = len;
memcpy( packet.buf, buf, len);
switch( ipc_priv.method_type )
{
case IPC_CALL_METHOD_FABIND :
{
ret = fabind_write( obj, packet.dst_id, (char *)(&packet), IPC_CALL_PACKET_HEADER_LEN + len );
}
break;
case IPC_CALL_METHOD_UDS :
{
char dst_uds[1024];
//memset(dst_uds, 0x00, sizeof(dst_uds));
//sprintf(dst_uds, "%s%ld", IPC_UDS_NAME_PREFIX, dest );
sprintf( dst_uds, IPC_CALL_UDS_NAME_FORMAT, IPC_CALL_GET_PORT(dest) );
ret = uds_write( obj, dst_uds, (char *)(&packet), IPC_CALL_PACKET_HEADER_LEN + len );
}
break;
case IPC_CALL_METHOD_UDP :
{
char host[1024];
//memset(host, 0x00, sizeof(host) );
sprintf(host, "%s", "127.0.0.1" );
ret = udp_write( obj, host, IPC_CALL_GET_PORT(dest), (char *)(&packet), IPC_CALL_PACKET_HEADER_LEN + len );
}
break;
default :
{
printf("Not Match Method\n");
}
break;
}
if ( 0 >= ret )
{
// todo
printf("ipc Send Fail\n");
return -1;
}
return 0;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 XML 형식으로 데이터를 전송한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_send_xml ( unsigned long dest, char *buf, int len )
{
return ipc_call_send( dest, buf, len, IPC_CALL_MSG_TYPE_XML );
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 ASC 형식으로 데이터를 전송한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 0 성공
-1 실패
*///----------------------------------------------------------------------------
int ipc_call_send_asc ( unsigned long dest, char *buf, int len )
{
return ipc_call_send( dest, buf, len, IPC_CALL_MSG_TYPE_ASC );
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 BIN 형식으로 데이터를 전송하고 수신한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 양수 성공
음수 실패
*///----------------------------------------------------------------------------
int ipc_call_rpc_raw ( unsigned long dest, void *tbuf, int tlen , void *rbuf, int rlen , unsigned long msg_type )
{
int ret = 0;
int event_ret = 0;
ipc_call_packet_t packet;
poll_obj_t *obj;
packet.dst_id = dest;
packet.src_id = ipc_priv.id;
packet.msg_type = msg_type;
packet.size = tlen;
memcpy( packet.buf, tbuf, tlen);
obj = ipc_priv.poll_obj;
switch( ipc_priv.method_type )
{
case IPC_CALL_METHOD_FABIND :
{
ret = fabind_write_and_response( obj, packet.dst_id, tbuf, rbuf, tlen, rlen );
}
break;
case IPC_CALL_METHOD_UDS :
{
char dst_uds[1024];
sprintf( dst_uds, IPC_CALL_UDS_NAME_FORMAT, IPC_CALL_GET_PORT(dest) );
ret = uds_write( obj, dst_uds, (char *)(&packet), IPC_CALL_PACKET_HEADER_LEN + tlen );
event_ret = poll_do_one( obj->fd, POLLIN, ipc_priv.time_out ); // POLL 이벤트가 올때까지 대기 한다. ( TIME OUT )이면 빠져 나간다.
if ( event_ret != POLL_EVENTED )
{
// 이벤트가 정상적으로 발생 할때 까지 무한 대기 한다.
while(1)
{
ret = uds_write( obj, dst_uds, (char *)(&packet), IPC_CALL_PACKET_HEADER_LEN + tlen );
if ( ( event_ret == POLL_TIME_OUT ) && ( ipc_priv.time_out == IPC_CALL_TIMEOUT_UNLIMIT ) )
{
event_ret = poll_do_one( obj->fd, POLLIN, ipc_priv.time_out ); // POLL 이벤트가 올때까지 대기 한다. ( TIME OUT )이면 빠져 나간다.
}
if ( event_ret == POLL_EVENTED )
{
break;
}
}
}
ret = uds_read( obj, rbuf, rlen );
}
break;
case IPC_CALL_METHOD_UDP :
{
char host[1024];
memset(host, 0x00, sizeof(host) );
sprintf(host, "%s", "127.0.0.1");
udp_write( obj, host, packet.dst_id, (char *)(&packet), tlen );
poll_do_one( obj->fd, POLLIN, ipc_priv.time_out ); // POLL 이벤트가 올때까지 대기 한다. ( TIME OUT )이면 빠져 나간다.
ret = udp_read( obj, rbuf, rlen );
}
break;
default :
{
printf("Not Match METHOD\n");
}
break;
}
if ( ret < 0 ) ipc_call_print_error( ret );
return ret;
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 BIN 형식으로 데이터를 전송하고 수신한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 양수 성공
음수 실패
*///----------------------------------------------------------------------------
int ipc_call_rpc( unsigned long dest, void *tbuf, int tlen , void *rbuf, int rlen )
{
return ipc_call_rpc_raw( dest, tbuf, tlen , rbuf, rlen , IPC_CALL_MSG_TYPE_BIN );
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 XML 형식으로 데이터를 전송하고 수신한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 양수 성공
음수 실패
*///----------------------------------------------------------------------------
int ipc_call_rpc_xml( unsigned long dest, void *tbuf, int tlen , void *rbuf, int rlen )
{
return ipc_call_rpc_raw( dest, tbuf, tlen , rbuf, rlen , IPC_CALL_MSG_TYPE_XML );
}
//------------------------------------------------------------------------------
/** @brief IPC_CALL 의 ASC 형식으로 데이터를 전송하고 수신한다.
@param msec timeout될 디폴트 시간을 정한다.
@return 양수 성공
음수 실패
*///----------------------------------------------------------------------------
int ipc_call_rpc_asc( unsigned long dest, void *tbuf, int tlen , void *rbuf, int rlen )
{
return ipc_call_rpc_raw( dest, tbuf, tlen , rbuf, rlen , IPC_CALL_MSG_TYPE_ASC );
}