author | Edouard Tisserant |
Tue, 20 Mar 2018 16:04:52 +0100 | |
changeset 800 | f0d16976e46e |
parent 793 | 72e9e1064432 |
child 801 | 32d146b64a35 |
permissions | -rw-r--r-- |
0 | 1 |
#include <stdlib.h> |
2 |
||
3 |
#include <sys/time.h> |
|
793
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
4 |
#include <pthread.h> |
0 | 5 |
#include <signal.h> |
631 | 6 |
#include <time.h> |
0 | 7 |
|
793
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
8 |
#include <applicfg.h> |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
9 |
#include <timer.h> |
0 | 10 |
|
631 | 11 |
static pthread_mutex_t CanFestival_mutex = PTHREAD_MUTEX_INITIALIZER; |
0 | 12 |
|
631 | 13 |
static struct timeval last_sig; |
32 | 14 |
|
631 | 15 |
static timer_t timer; |
0 | 16 |
|
454 | 17 |
void TimerCleanup(void) |
18 |
{ |
|
19 |
/* only used in realtime apps */ |
|
20 |
} |
|
21 |
||
0 | 22 |
void EnterMutex(void) |
23 |
{ |
|
631 | 24 |
if(pthread_mutex_lock(&CanFestival_mutex)) { |
25 |
fprintf(stderr, "pthread_mutex_lock() failed\n"); |
|
26 |
} |
|
0 | 27 |
} |
28 |
||
29 |
void LeaveMutex(void) |
|
30 |
{ |
|
631 | 31 |
if(pthread_mutex_unlock(&CanFestival_mutex)) { |
32 |
fprintf(stderr, "pthread_mutex_unlock() failed\n"); |
|
33 |
} |
|
0 | 34 |
} |
35 |
||
145
e747d2e26af0
Win32 Native support and dynamicaly loaded CAN drivers for Linux, Cygwin and Win32.
etisserant
parents:
48
diff
changeset
|
36 |
void timer_notify(sigval_t val) |
32 | 37 |
{ |
631 | 38 |
if(gettimeofday(&last_sig,NULL)) { |
39 |
perror("gettimeofday()"); |
|
40 |
} |
|
32 | 41 |
EnterMutex(); |
42 |
TimeDispatch(); |
|
43 |
LeaveMutex(); |
|
44 |
// printf("getCurrentTime() return=%u\n", p.tv_usec); |
|
45 |
} |
|
46 |
||
454 | 47 |
void TimerInit(void) |
32 | 48 |
{ |
49 |
struct sigevent sigev; |
|
50 |
||
51 |
// Take first absolute time ref. |
|
631 | 52 |
if(gettimeofday(&last_sig,NULL)){ |
53 |
perror("gettimeofday()"); |
|
54 |
} |
|
32 | 55 |
|
607
5fec528f66cf
create specific timer thread for UCLIBC support (with CLOCK_PROCESS_CPUTIME_ID parameter)
greg
parents:
540
diff
changeset
|
56 |
#if defined(__UCLIBC__) |
5fec528f66cf
create specific timer thread for UCLIBC support (with CLOCK_PROCESS_CPUTIME_ID parameter)
greg
parents:
540
diff
changeset
|
57 |
int ret; |
5fec528f66cf
create specific timer thread for UCLIBC support (with CLOCK_PROCESS_CPUTIME_ID parameter)
greg
parents:
540
diff
changeset
|
58 |
ret = timer_create(CLOCK_PROCESS_CPUTIME_ID, NULL, &timer); |
5fec528f66cf
create specific timer thread for UCLIBC support (with CLOCK_PROCESS_CPUTIME_ID parameter)
greg
parents:
540
diff
changeset
|
59 |
signal(SIGALRM, timer_notify); |
5fec528f66cf
create specific timer thread for UCLIBC support (with CLOCK_PROCESS_CPUTIME_ID parameter)
greg
parents:
540
diff
changeset
|
60 |
#else |
32 | 61 |
memset (&sigev, 0, sizeof (struct sigevent)); |
62 |
sigev.sigev_value.sival_int = 0; |
|
63 |
sigev.sigev_notify = SIGEV_THREAD; |
|
64 |
sigev.sigev_notify_attributes = NULL; |
|
65 |
sigev.sigev_notify_function = timer_notify; |
|
66 |
||
631 | 67 |
if(timer_create (CLOCK_REALTIME, &sigev, &timer)) { |
68 |
perror("timer_create()"); |
|
69 |
} |
|
607
5fec528f66cf
create specific timer thread for UCLIBC support (with CLOCK_PROCESS_CPUTIME_ID parameter)
greg
parents:
540
diff
changeset
|
70 |
#endif |
32 | 71 |
} |
72 |
||
454 | 73 |
void StopTimerLoop(TimerCallback_t exitfunction) |
32 | 74 |
{ |
149 | 75 |
EnterMutex(); |
631 | 76 |
if(timer_delete (timer)) { |
77 |
perror("timer_delete()"); |
|
78 |
} |
|
454 | 79 |
exitfunction(NULL,0); |
149 | 80 |
LeaveMutex(); |
32 | 81 |
} |
82 |
||
83 |
void StartTimerLoop(TimerCallback_t init_callback) |
|
0 | 84 |
{ |
149 | 85 |
EnterMutex(); |
0 | 86 |
// At first, TimeDispatch will call init_callback. |
87 |
SetAlarm(NULL, 0, init_callback, 0, 0); |
|
149 | 88 |
LeaveMutex(); |
0 | 89 |
} |
90 |
||
507
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
91 |
void canReceiveLoop_signal(int sig) |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
92 |
{ |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
93 |
} |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
94 |
/* We assume that ReceiveLoop_task_proc is always the same */ |
508
08adb8d4b098
Added more correct signal handling to unix timers.
etisserant
parents:
507
diff
changeset
|
95 |
static void (*unixtimer_ReceiveLoop_task_proc)(CAN_PORT) = NULL; |
507
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
96 |
|
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
97 |
/** |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
98 |
* Enter in realtime and start the CAN receiver loop |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
99 |
* @param port |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
100 |
*/ |
540 | 101 |
void* unixtimer_canReceiveLoop(void* port) |
507
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
102 |
{ |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
103 |
/*get signal*/ |
663 | 104 |
// if(signal(SIGTERM, canReceiveLoop_signal) == SIG_ERR) { |
105 |
// perror("signal()"); |
|
106 |
//} |
|
793
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
107 |
|
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
108 |
// Set the cancelation state for immediatly cancel the task |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
109 |
int ret; |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
110 |
|
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
111 |
ret = pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL); |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
112 |
|
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
113 |
if (ret != 0) |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
114 |
perror("Can't enable the cancelation of the receiving task"); |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
115 |
|
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
116 |
ret = pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, NULL); |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
117 |
|
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
118 |
if (ret != 0) |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
119 |
perror("Can't set the asynchronous cancel typ"); |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
120 |
|
540 | 121 |
unixtimer_ReceiveLoop_task_proc((CAN_PORT)port); |
631 | 122 |
|
123 |
return NULL; |
|
507
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
124 |
} |
c613e6cd34fa
Added more correct signal handling to unix timers.
etisserant
parents:
454
diff
changeset
|
125 |
|
145
e747d2e26af0
Win32 Native support and dynamicaly loaded CAN drivers for Linux, Cygwin and Win32.
etisserant
parents:
48
diff
changeset
|
126 |
void CreateReceiveTask(CAN_PORT port, TASK_HANDLE* Thread, void* ReceiveLoopPtr) |
0 | 127 |
{ |
508
08adb8d4b098
Added more correct signal handling to unix timers.
etisserant
parents:
507
diff
changeset
|
128 |
unixtimer_ReceiveLoop_task_proc = ReceiveLoopPtr; |
631 | 129 |
if(pthread_create(Thread, NULL, unixtimer_canReceiveLoop, (void*)port)) { |
130 |
perror("pthread_create()"); |
|
131 |
} |
|
0 | 132 |
} |
133 |
||
401
2c90876b9751
Fixed segfault on quit with Xenomai, due to bat parameter type in waitReceiveTaskEnd.
etisserant
parents:
149
diff
changeset
|
134 |
void WaitReceiveTaskEnd(TASK_HANDLE *Thread) |
0 | 135 |
{ |
793
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
136 |
if(pthread_cancel(*Thread)) { |
72e9e1064432
timers_unix: Fix termination problem of WaitReceiveTaskEnd
Robert Lehmann <robert.lehmann@sitec-systems.de>
parents:
663
diff
changeset
|
137 |
perror("pthread_cancel()"); |
631 | 138 |
} |
139 |
if(pthread_join(*Thread, NULL)) { |
|
140 |
perror("pthread_join()"); |
|
141 |
} |
|
0 | 142 |
} |
143 |
||
48
adc6572caf5d
minval/maxval macro operators precedence fix. Thanks Luis Jim?nez.
etisserant
parents:
38
diff
changeset
|
144 |
#define maxval(a,b) ((a>b)?a:b) |
0 | 145 |
void setTimer(TIMEVAL value) |
146 |
{ |
|
147 |
// printf("setTimer(TIMEVAL value=%d)\n", value); |
|
32 | 148 |
// TIMEVAL is us whereas setitimer wants ns... |
38 | 149 |
long tv_nsec = 1000 * (maxval(value,1)%1000000); |
32 | 150 |
time_t tv_sec = value/1000000; |
151 |
struct itimerspec timerValues; |
|
152 |
timerValues.it_value.tv_sec = tv_sec; |
|
153 |
timerValues.it_value.tv_nsec = tv_nsec; |
|
0 | 154 |
timerValues.it_interval.tv_sec = 0; |
32 | 155 |
timerValues.it_interval.tv_nsec = 0; |
156 |
||
157 |
timer_settime (timer, 0, &timerValues, NULL); |
|
0 | 158 |
} |
159 |
||
160 |
TIMEVAL getElapsedTime(void) |
|
161 |
{ |
|
162 |
struct timeval p; |
|
631 | 163 |
if(gettimeofday(&p,NULL)) { |
164 |
perror("gettimeofday()"); |
|
165 |
} |
|
0 | 166 |
// printf("getCurrentTime() return=%u\n", p.tv_usec); |
167 |
return (p.tv_sec - last_sig.tv_sec)* 1000000 + p.tv_usec - last_sig.tv_usec; |
|
168 |
} |