Передача пользовательских данных с помощью timer_create - PullRequest
3 голосов
/ 10 мая 2011

Я использую timer_create для создания таймера в Linux.Прототип обратного вызова:

static void TimerHandlerCB(int sig, siginfo_t *extra, void *cruft)

Как я могу передать пользовательские данные, чтобы я мог получить то же самое в обратном вызове, вызванном после истечения таймера.

Вот мой пример кода:

int RegisterTimer(iDiffInTime)
{
    struct sigaction sa; 
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = SA_SIGINFO;   /*call our handler*/
    sa.sa_sigaction = TimerHandlerCB;/*Event handler to be called after timer expires*/ 
    if(sigaction(SIGRTMAX, &sa, NULL) < 0)
    {
        perror("sigaction");
        return 1;
    }
    // Setup the timer
    sigevent sigev;
    timer_t timerid = 0;

    memset(&sigev, 0, sizeof(sigev));

    sigev.sigev_notify          = SIGEV_SIGNAL;
    sigev.sigev_signo           = SIGRTMAX;
    sigev.sigev_value.sival_ptr = &timerid;

    timer_t timerid;

    if (timer_create(CLOCK_REALTIME, &sigev, &timerid) == 0)
    {
        printf("Timer created\n");
        struct itimerspec itval, oitval;

        itval.it_value.tv_sec = iDiffInTime * 1000;
        itval.it_value.tv_nsec = 0;
        itval.it_interval.tv_sec = 0;
        itval.it_interval.tv_nsec = 0;

              //Populate handles required in TimerCallback
              Display_Handle hDisplay = ......//
              Buffer_Handle hBuf = .....//
        if (timer_settime(timerid, 0, &itval, &oitval) != 0)
        {
            perror("time_settime error!");
            return 1;
        }
    } 
    else 
    {
        perror("timer_create error!");
    }
    return 0
}

Где я могу передать hDisplay & hBuf, чтобы я мог получить их обратно в TimerHandlerCB?

Ответы [ 2 ]

3 голосов
/ 10 мая 2011

Вы уже делаете это:

sigev.sigev_value.sival_ptr = &timerid;

Здесь вы можете указать что угодно (указание &timerid является общим для различения между несколькими таймерами, но не является обязательным).Теперь в вашем обработчике:

static void timer_handler(int signo, siginfo_t *si, void *uc)
{
    /* si->si_value.sival_ptr */
}

Цитата из TLPI

При вызове timer_create () evp.sigev_value.sival_ptr обычно присваивается адрес аргумента timerid, указанного в том же самомвызов.В качестве альтернативы, evp.sigev_value.sival_ptr может быть назначен адрес структуры, которая содержит таймер, данный для timer_create ()

1 голос
/ 10 мая 2011

Вам необходимо:

  • добавить глобальную переменную и использовать какой-нибудь механизм синхронизации, например семафоры
  • использовать sigqueue
...