xref: /OK3568_Linux_fs/external/mpp/test/mpp_event_trigger.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright 2017 Rockchip Electronics Co. LTD
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Licensed under the Apache License, Version 2.0 (the "License");
5*4882a593Smuzhiyun  * you may not use this file except in compliance with the License.
6*4882a593Smuzhiyun  * You may obtain a copy of the License at
7*4882a593Smuzhiyun  *
8*4882a593Smuzhiyun  *      http://www.apache.org/licenses/LICENSE-2.0
9*4882a593Smuzhiyun  *
10*4882a593Smuzhiyun  * Unless required by applicable law or agreed to in writing, software
11*4882a593Smuzhiyun  * distributed under the License is distributed on an "AS IS" BASIS,
12*4882a593Smuzhiyun  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4882a593Smuzhiyun  * See the License for the specific language governing permissions and
14*4882a593Smuzhiyun  * limitations under the License.
15*4882a593Smuzhiyun  */
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #include <assert.h>
18*4882a593Smuzhiyun #include <errno.h>
19*4882a593Smuzhiyun #include <malloc.h>
20*4882a593Smuzhiyun #include <memory.h>
21*4882a593Smuzhiyun #include <pthread.h>
22*4882a593Smuzhiyun #include <stdio.h>
23*4882a593Smuzhiyun #include <sys/time.h>
24*4882a593Smuzhiyun #include <unistd.h>
25*4882a593Smuzhiyun 
26*4882a593Smuzhiyun #include "mpp_event_trigger.h"
27*4882a593Smuzhiyun 
28*4882a593Smuzhiyun #include "mpp_debug.h"
29*4882a593Smuzhiyun 
30*4882a593Smuzhiyun struct event_ctx_impl {
31*4882a593Smuzhiyun     int (*notify)(void *param);
32*4882a593Smuzhiyun 
33*4882a593Smuzhiyun     event_trigger trigger;
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun     pthread_cond_t condition;
36*4882a593Smuzhiyun     pthread_mutex_t mutex;
37*4882a593Smuzhiyun     RK_U32 semval;
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun     pthread_t thr;
40*4882a593Smuzhiyun     RK_S32 flag;
41*4882a593Smuzhiyun 
42*4882a593Smuzhiyun     struct event_packet *ea;
43*4882a593Smuzhiyun     int event_idx;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun     void *parent;
46*4882a593Smuzhiyun };
47*4882a593Smuzhiyun 
event_create(struct event_ctx_impl * ctx)48*4882a593Smuzhiyun static int event_create(struct event_ctx_impl *ctx)
49*4882a593Smuzhiyun {
50*4882a593Smuzhiyun     int ret;
51*4882a593Smuzhiyun 
52*4882a593Smuzhiyun     ret = pthread_cond_init(&ctx->condition, NULL);
53*4882a593Smuzhiyun     if (ret != 0)
54*4882a593Smuzhiyun         return -1;
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun     ret = pthread_mutex_init(&ctx->mutex, NULL);
57*4882a593Smuzhiyun     if (ret != 0)
58*4882a593Smuzhiyun         return -1;
59*4882a593Smuzhiyun 
60*4882a593Smuzhiyun     ctx->semval = ctx->ea->e[0].idx;
61*4882a593Smuzhiyun     mpp_log_f("with %u\n", ctx->semval);
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun     return 0;
64*4882a593Smuzhiyun }
65*4882a593Smuzhiyun 
event_destroy(struct event_ctx_impl * ctx)66*4882a593Smuzhiyun static void event_destroy(struct event_ctx_impl *ctx)
67*4882a593Smuzhiyun {
68*4882a593Smuzhiyun     pthread_cond_destroy(&ctx->condition);
69*4882a593Smuzhiyun     pthread_mutex_destroy(&ctx->mutex);
70*4882a593Smuzhiyun }
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun /* initialize and schedule next event */
event_init(struct event_ctx_impl * ctx)73*4882a593Smuzhiyun static void event_init(struct event_ctx_impl *ctx)
74*4882a593Smuzhiyun {
75*4882a593Smuzhiyun     struct ievent *e_curr;
76*4882a593Smuzhiyun     struct ievent *e_next;
77*4882a593Smuzhiyun 
78*4882a593Smuzhiyun     pthread_mutex_lock(&ctx->mutex);
79*4882a593Smuzhiyun     e_curr = &ctx->ea->e[ctx->event_idx % ctx->ea->cnt];
80*4882a593Smuzhiyun     e_next = &ctx->ea->e[(++ctx->event_idx) % ctx->ea->cnt];
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun     mpp_log("curr %d, next %d\n",
83*4882a593Smuzhiyun             e_curr->idx, e_next->idx);
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun     if (e_next->idx > e_curr->idx)
86*4882a593Smuzhiyun         ctx->semval = e_next->idx - e_curr->idx;
87*4882a593Smuzhiyun     else if (ctx->ea->loop > 0)
88*4882a593Smuzhiyun         ctx->semval = e_next->idx + ctx->ea->loop - e_curr->idx;
89*4882a593Smuzhiyun     else
90*4882a593Smuzhiyun         ctx->flag = 0;
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun     mpp_log_f("semval %u\n", ctx->semval);
93*4882a593Smuzhiyun     pthread_mutex_unlock(&ctx->mutex);
94*4882a593Smuzhiyun }
95*4882a593Smuzhiyun 
96*4882a593Smuzhiyun /* wait the event trigger deadline */
97*4882a593Smuzhiyun /*static void event_wait(struct event_ctx_impl *ctx)
98*4882a593Smuzhiyun {
99*4882a593Smuzhiyun     pthread_mutex_lock(&ctx->mutex);
100*4882a593Smuzhiyun     while (ctx->semval > 0)
101*4882a593Smuzhiyun         pthread_cond_wait(&ctx->condition, &ctx->mutex);
102*4882a593Smuzhiyun 
103*4882a593Smuzhiyun     ctx->semval--;
104*4882a593Smuzhiyun     pthread_mutex_unlock(&ctx->mutex);
105*4882a593Smuzhiyun }*/
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun /* wait the event trigger deadline with timeout, avoid dead lock */
event_timed_wait(struct event_ctx_impl * ctx,unsigned int milli_sec)108*4882a593Smuzhiyun static int event_timed_wait(struct event_ctx_impl *ctx, unsigned int milli_sec)
109*4882a593Smuzhiyun {
110*4882a593Smuzhiyun     int err = 0;
111*4882a593Smuzhiyun     struct timespec final_time;
112*4882a593Smuzhiyun     struct timeval curr_time;
113*4882a593Smuzhiyun     long int microdelay;
114*4882a593Smuzhiyun 
115*4882a593Smuzhiyun     gettimeofday(&curr_time, NULL);
116*4882a593Smuzhiyun     /*
117*4882a593Smuzhiyun      * convert timeval to timespec and add delay in milliseconds
118*4882a593Smuzhiyun      * for the timeout
119*4882a593Smuzhiyun      */
120*4882a593Smuzhiyun     microdelay = ((milli_sec * 1000 + curr_time.tv_usec));
121*4882a593Smuzhiyun     final_time.tv_sec = curr_time.tv_sec + (microdelay / 1000000);
122*4882a593Smuzhiyun     final_time.tv_nsec = (microdelay % 1000000) * 1000;
123*4882a593Smuzhiyun     pthread_mutex_lock(&ctx->mutex);
124*4882a593Smuzhiyun     while (ctx->semval > 0) {
125*4882a593Smuzhiyun         err = pthread_cond_timedwait(&ctx->condition, &ctx->mutex, &final_time);
126*4882a593Smuzhiyun         if (err != 0)
127*4882a593Smuzhiyun             ctx->semval--;
128*4882a593Smuzhiyun     }
129*4882a593Smuzhiyun     ctx->semval--;
130*4882a593Smuzhiyun     pthread_mutex_unlock(&ctx->mutex);
131*4882a593Smuzhiyun     return err;
132*4882a593Smuzhiyun }
133*4882a593Smuzhiyun 
134*4882a593Smuzhiyun /* event heart beat */
event_down(struct event_ctx_impl * ctx)135*4882a593Smuzhiyun static void event_down(struct event_ctx_impl *ctx)
136*4882a593Smuzhiyun {
137*4882a593Smuzhiyun     pthread_mutex_lock(&ctx->mutex);
138*4882a593Smuzhiyun     ctx->semval--;
139*4882a593Smuzhiyun     pthread_cond_signal(&ctx->condition);
140*4882a593Smuzhiyun     pthread_mutex_unlock(&ctx->mutex);
141*4882a593Smuzhiyun }
142*4882a593Smuzhiyun 
143*4882a593Smuzhiyun /* a callback to notify the event trigger that there is a event heart beat */
event_notify(void * param)144*4882a593Smuzhiyun static int event_notify(void *param)
145*4882a593Smuzhiyun {
146*4882a593Smuzhiyun     struct event_ctx_impl *ctx = (struct event_ctx_impl *)param;
147*4882a593Smuzhiyun 
148*4882a593Smuzhiyun     if (ctx->flag)
149*4882a593Smuzhiyun         event_down(ctx);
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun     return 0;
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun 
event_trigger_thread(void * param)154*4882a593Smuzhiyun static void *event_trigger_thread(void *param)
155*4882a593Smuzhiyun {
156*4882a593Smuzhiyun     struct event_ctx_impl *ctx = (struct event_ctx_impl *)param;
157*4882a593Smuzhiyun     struct timeval curr_time;
158*4882a593Smuzhiyun     long start_time, curr;
159*4882a593Smuzhiyun     RK_S32 ret;
160*4882a593Smuzhiyun 
161*4882a593Smuzhiyun     gettimeofday(&curr_time, NULL);
162*4882a593Smuzhiyun 
163*4882a593Smuzhiyun     start_time = curr_time.tv_sec * 1000 + curr_time.tv_usec / 1000;
164*4882a593Smuzhiyun     event_create(ctx);
165*4882a593Smuzhiyun 
166*4882a593Smuzhiyun     while (1) {
167*4882a593Smuzhiyun         ret = event_timed_wait(ctx, ctx->semval * 10000);
168*4882a593Smuzhiyun         if (!ctx->flag)
169*4882a593Smuzhiyun             break;
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun         if (ret == ETIMEDOUT) {
172*4882a593Smuzhiyun             mpp_err("wait event timeout\n");
173*4882a593Smuzhiyun             break;
174*4882a593Smuzhiyun         }
175*4882a593Smuzhiyun 
176*4882a593Smuzhiyun         gettimeofday(&curr_time, NULL);
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun         curr = curr_time.tv_sec * 1000 + curr_time.tv_usec / 1000;
179*4882a593Smuzhiyun         /* TODO, trigger event */
180*4882a593Smuzhiyun         mpp_log("[%ld ms] event idx %d triggered\n",
181*4882a593Smuzhiyun                 curr - start_time, ctx->event_idx);
182*4882a593Smuzhiyun         mpp_log("cnt %d\n", ctx->ea->cnt);
183*4882a593Smuzhiyun         ctx->trigger(ctx->parent, ctx->ea->e[ctx->event_idx % ctx->ea->cnt].event);
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun         /* schedule next event */
186*4882a593Smuzhiyun         event_init(ctx);
187*4882a593Smuzhiyun         if (!ctx->flag)
188*4882a593Smuzhiyun             break;
189*4882a593Smuzhiyun     }
190*4882a593Smuzhiyun 
191*4882a593Smuzhiyun     event_destroy(ctx);
192*4882a593Smuzhiyun 
193*4882a593Smuzhiyun     mpp_log_f("exit\n");
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun     return NULL;
196*4882a593Smuzhiyun }
197*4882a593Smuzhiyun 
event_ctx_create(struct event_packet * ea,event_trigger trigger,void * parent)198*4882a593Smuzhiyun struct event_ctx* event_ctx_create(struct event_packet *ea,
199*4882a593Smuzhiyun                                    event_trigger trigger, void *parent)
200*4882a593Smuzhiyun {
201*4882a593Smuzhiyun     struct event_ctx_impl *ctx = (struct event_ctx_impl *)malloc(sizeof(*ctx));
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun     if (ctx == NULL) {
204*4882a593Smuzhiyun         mpp_err("allocate event ctx failed\n");
205*4882a593Smuzhiyun         return NULL;
206*4882a593Smuzhiyun     }
207*4882a593Smuzhiyun 
208*4882a593Smuzhiyun     assert(ea->cnt <= 128);
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun     ctx->event_idx = 0;
211*4882a593Smuzhiyun     ctx->ea = ea;
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun     ctx->notify = event_notify;
214*4882a593Smuzhiyun     ctx->trigger = trigger;
215*4882a593Smuzhiyun     ctx->parent = parent;
216*4882a593Smuzhiyun 
217*4882a593Smuzhiyun     ctx->flag = 1;
218*4882a593Smuzhiyun     pthread_create(&ctx->thr, NULL, event_trigger_thread, ctx);
219*4882a593Smuzhiyun 
220*4882a593Smuzhiyun     return (struct event_ctx *)ctx;
221*4882a593Smuzhiyun }
222*4882a593Smuzhiyun 
event_ctx_release(struct event_ctx * ctx)223*4882a593Smuzhiyun void event_ctx_release(struct event_ctx *ctx)
224*4882a593Smuzhiyun {
225*4882a593Smuzhiyun     void *ret;
226*4882a593Smuzhiyun     struct event_ctx_impl *ictx = (struct event_ctx_impl *)ctx;
227*4882a593Smuzhiyun 
228*4882a593Smuzhiyun     assert(ctx != NULL);
229*4882a593Smuzhiyun 
230*4882a593Smuzhiyun     if (ictx->flag) {
231*4882a593Smuzhiyun         ictx->flag = 0;
232*4882a593Smuzhiyun         ictx->semval = 0;
233*4882a593Smuzhiyun         pthread_cond_signal(&ictx->condition);
234*4882a593Smuzhiyun     }
235*4882a593Smuzhiyun 
236*4882a593Smuzhiyun     pthread_join(ictx->thr, &ret);
237*4882a593Smuzhiyun 
238*4882a593Smuzhiyun     free(ictx);
239*4882a593Smuzhiyun }
240*4882a593Smuzhiyun 
241*4882a593Smuzhiyun #ifdef EVENT_TRIGGER_TEST
242*4882a593Smuzhiyun /*
243*4882a593Smuzhiyun  * the following codes is the sample to use the event trigger
244*4882a593Smuzhiyun  */
245*4882a593Smuzhiyun struct event_impl {
246*4882a593Smuzhiyun     int cnt;
247*4882a593Smuzhiyun     int loop;
248*4882a593Smuzhiyun     int idx[128];
249*4882a593Smuzhiyun     int event[128];
250*4882a593Smuzhiyun };
251*4882a593Smuzhiyun 
252*4882a593Smuzhiyun /* a callback to notify test that a event occur */
event_occur(void * parent,void * event)253*4882a593Smuzhiyun static void event_occur(void *parent, void *event)
254*4882a593Smuzhiyun {
255*4882a593Smuzhiyun     int *e = (int *)event;
256*4882a593Smuzhiyun     mpp_log("event %d occur\n", *e);
257*4882a593Smuzhiyun }
258*4882a593Smuzhiyun 
main(int argc,char ** argv)259*4882a593Smuzhiyun int main(int argc, char **argv)
260*4882a593Smuzhiyun {
261*4882a593Smuzhiyun     struct event_ctx *ctx;
262*4882a593Smuzhiyun     int i = 0;
263*4882a593Smuzhiyun     struct event_impl ie;
264*4882a593Smuzhiyun     struct event_packet ea;
265*4882a593Smuzhiyun 
266*4882a593Smuzhiyun     ie.cnt = 4;
267*4882a593Smuzhiyun     ie.loop = 25;
268*4882a593Smuzhiyun     ie.event[0] = ie.idx[0] = 4;
269*4882a593Smuzhiyun     ie.event[1] = ie.idx[1] = 8;
270*4882a593Smuzhiyun     ie.event[2] = ie.idx[2] = 18;
271*4882a593Smuzhiyun     ie.event[3] = ie.idx[3] = 20;
272*4882a593Smuzhiyun 
273*4882a593Smuzhiyun     ea.cnt = ie.cnt;
274*4882a593Smuzhiyun     ea.loop = ie.loop;
275*4882a593Smuzhiyun 
276*4882a593Smuzhiyun     for (i = 0; i < ie.cnt; ++i) {
277*4882a593Smuzhiyun         ea.e[i].idx = ie.idx[i];
278*4882a593Smuzhiyun         ea.e[i].event = &ie.event[i];
279*4882a593Smuzhiyun     }
280*4882a593Smuzhiyun 
281*4882a593Smuzhiyun     ctx = event_ctx_create(&ea, event_occur, NULL);
282*4882a593Smuzhiyun 
283*4882a593Smuzhiyun     while (i++ < 100) {
284*4882a593Smuzhiyun         usleep(10 * 1000);
285*4882a593Smuzhiyun         mpp_log("%03d:\n", i);
286*4882a593Smuzhiyun         ctx->notify((void*)ctx);
287*4882a593Smuzhiyun     }
288*4882a593Smuzhiyun 
289*4882a593Smuzhiyun     event_ctx_release(ctx);
290*4882a593Smuzhiyun 
291*4882a593Smuzhiyun     return 0;
292*4882a593Smuzhiyun }
293*4882a593Smuzhiyun #endif
294