linux timerfd系列函數總結

網上關於timerfd的文章不少,在這兒概括總結一下方便之後使用,順便貼出一個timerfd配合epoll使用的簡單例子html

1、timerfd系列函數linux

  timerfd是Linux爲用戶程序提供的一個定時器接口。這個接口基於文件描述符,經過文件描述符的可讀事件進行超時通知,所以能夠配合select/poll/epoll等使用。
下面對timerfd系列函數先作一個簡單的介紹:函數

(1)timerfd_create()函數ui

#include <sys/timerfd.h>

int timerfd_create(int clockid, int flags);
/* 
timerfd_create()函數建立一個定時器對象,同時返回一個與之關聯的文件描述符。
clockid:clockid標識指定的時鐘計數器,可選值(CLOCK_REALTIME、CLOCK_MONOTONIC。。。)
CLOCK_REALTIME:系統實時時間,隨系統實時時間改變而改變,即從UTC1970-1-1 0:0:0開始計時,中間時刻若是系統時間被用戶改爲其餘,則對應的時間相應改變
CLOCK_MONOTONIC:從系統啓動這一刻起開始計時,不受系統時間被用戶改變的影響
flags:參數flags(TFD_NONBLOCK(非阻塞模式)/TFD_CLOEXEC(表示當程序執行exec函數時本fd將被系統自動關閉,表示不傳遞)
*/

(2)timerfd_settime()函數spa

 1 #include <sys/timerfd.h>
 2 
 3 struct timespec {
 4     time_t tv_sec;                /* Seconds */
 5     long   tv_nsec;               /* Nanoseconds */
 6 };
 7 
 8 struct itimerspec {
 9     struct timespec it_interval;  /* Interval for periodic timer (定時間隔週期)*/
10     struct timespec it_value;     /* Initial expiration (第一次超時時間)*/
11 };
12 int timerfd_settime(int fd, int flags, const struct itimerspec *new_value, struct itimerspec *old_value);
13 /*
14     timerfd_settime()此函數用於設置新的超時時間,並開始計時,可以啓動和中止定時器;
15     fd: 參數fd是timerfd_create函數返回的文件句柄
16     flags:參數flags爲1表明設置的是絕對時間(TFD_TIMER_ABSTIME 表示絕對定時器);爲0表明相對時間。
17     new_value: 參數new_value指定定時器的超時時間以及超時間隔時間
18     old_value: 若是old_value不爲NULL, old_vlaue返回以前定時器設置的超時時間,具體參考timerfd_gettime()函數
19     
20     ** it_interval不爲0則表示是週期性定時器。
21        it_value和it_interval都爲0表示中止定時器
22 */

(3)timerfd_gettime()函數.net

1 int timerfd_gettime(int fd, struct itimerspec *curr_value);
2 /*
3     timerfd_gettime()函數獲取距離下次超時剩餘的時間
4     curr_value.it_value 字段表示距離下次超時的時間,若是改值爲0,表示計時器已經解除
5     改字段表示的值永遠是一個相對值,不管TFD_TIMER_ABSTIME是否被設置
6     curr_value.it_interval 定時器間隔時間
7 */
1 uint64_t exp = 0;
2 read(fd, &exp, sizeof(uint64_t)); 
3 //能夠用read函數讀取計時器的超時次數,改值是一個8字節無符號的長整型

(4)下面貼出一個timerfd配合epoll函數的簡單例子3d

  1 /********************************************************
  2 * Filename: timerfd.c
  3 * Author: zhangwj
  4 * Desprition: a sample program of timerfd
  5 * Date: 2017-04-17
  6 * Warnning:
  7 ********************************************************/
  8 #include <stdio.h>
  9 #include <stdint.h>
 10 #include <string.h>
 11 #include <stdlib.h>
 12 #include <pthread.h>
 13 #include <errno.h>
 14 #include <sys/epoll.h>
 15 #include <sys/timerfd.h>
 16 
 17 #if 0
 18 struct timespec {
 19     time_t tv_sec;                /* Seconds */
 20     long   tv_nsec;               /* Nanoseconds */
 21 };
 22 
 23 struct itimerspec {
 24     struct timespec it_interval;  /* Interval for periodic timer */
 25     struct timespec it_value;     /* Initial expiration */
 26 };
 27 #endif
 28 
 29 #define EPOLL_LISTEN_CNT        256
 30 #define EPOLL_LISTEN_TIMEOUT    500
 31 
 32 #define LOG_DEBUG_ON 1
 33 
 34 #ifdef LOG_DEBUG_ON 
 35 #define LOG_DEBUG(fmt, args...) \
 36     do {  \
 37         printf("[DEBUG]:");\
 38         printf(fmt "\n", ##args); \
 39     } while(0);
 40 #define LOG_INFO(fmt, args...) \
 41     do { \
 42         printf("[INFO]:");\
 43         printf(fmt "\n", ##args); \
 44     } while(0);
 45 #define LOG_WARNING(fmt, args...) \
 46     do { \
 47         printf("[WARNING]:");\
 48         printf(fmt "\n", ##args); \
 49     } while(0);
 50 #else
 51 #define LOG_DEBUG(fmt, args...) 
 52 #define LOG_INFO(fmt, args...) 
 53 #define LOG_WARNING(fmt, args...) 
 54 #endif
 55 #define LOG_ERROR(fmt, args...) \
 56     do{ \
 57         printf("[ERROR]:");\
 58         printf(fmt "\n", ##args);\
 59     }while(0);
 60 
 61 #define handle_error(msg) \
 62         do { perror(msg); exit(EXIT_FAILURE); } while (0)
 63 
 64 static int g_epollfd = -1;
 65 static int g_timerfd = -1;
 66 uint64_t tot_exp = 0;
 67 
 68 static void help(void)
 69 {
 70     exit(0);
 71 }
 72 
 73 static void print_elapsed_time(void)
 74 {
 75     static struct timespec start;
 76     struct timespec curr;
 77     static int first_call = 1;
 78     int secs, nsecs;
 79     
 80     if (first_call) {
 81         first_call = 0;
 82         if (clock_gettime(CLOCK_MONOTONIC, &start) == -1) 
 83             handle_error("clock_gettime");
 84     }   
 85     
 86     if (clock_gettime(CLOCK_MONOTONIC, &curr) == -1) 
 87         handle_error("clock_gettime");
 88     
 89     secs = curr.tv_sec - start.tv_sec;
 90     nsecs = curr.tv_nsec - start.tv_nsec;
 91     if (nsecs < 0) {
 92         secs--;
 93         nsecs += 1000000000;
 94     }   
 95     printf("%d.%03d: ", secs, (nsecs + 500000) / 1000000);
 96 }
 97 
 98 void timerfd_handler(int fd)
 99 {
100     uint64_t exp = 0;
101     
102     read(fd, &exp, sizeof(uint64_t)); 
103     tot_exp += exp;
104     print_elapsed_time();
105     printf("read: %llu, total: %llu\n", (unsigned long long)exp, (unsigned long long)tot_exp);
106 
107     return;
108 }
109 
110 void epoll_event_handle(void)
111 {
112     int i = 0;
113     int fd_cnt = 0;
114     int sfd;
115     struct epoll_event events[EPOLL_LISTEN_CNT];    
116 
117     memset(events, 0, sizeof(events));
118     while(1) 
119     {   
120         /* wait epoll event */
121         fd_cnt = epoll_wait(g_epollfd, events, EPOLL_LISTEN_CNT, EPOLL_LISTEN_TIMEOUT); 
122         for(i = 0; i < fd_cnt; i++) 
123         {   
124             sfd = events[i].data.fd;
125             if(events[i].events & EPOLLIN) 
126             {   
127                 if (sfd == g_timerfd) 
128                 {
129                     timerfd_handler(sfd); 
130                 }   
131             }   
132         } 
133     }   
134 }
135 
136 int epoll_add_fd(int fd)
137 {
138     int ret;
139     struct epoll_event event;
140 
141     memset(&event, 0, sizeof(event));
142     event.data.fd = fd;
143     event.events = EPOLLIN | EPOLLET;
144 
145     ret = epoll_ctl(g_epollfd, EPOLL_CTL_ADD, fd, &event);
146     if(ret < 0) {
147         LOG_ERROR("epoll_ctl Add fd:%d error, Error:[%d:%s]", fd, errno, strerror(errno));
148         return -1;
149     }
150 
151     LOG_DEBUG("epoll add fd:%d--->%d success", fd, g_epollfd);
152     return 0;    
153 }
154 
155 int epollfd_init()
156 {
157     int epfd;
158 
159     /* create epoll fd */
160     epfd = epoll_create(EPOLL_LISTEN_CNT); 
161     if (epfd < 0) {
162         LOG_ERROR("epoll_create error, Error:[%d:%s]", errno, strerror(errno));
163         return -1;
164     }
165     g_epollfd = epfd;
166     LOG_DEBUG("epoll fd:%d create success", epfd);
167 
168     return epfd;
169 }
170 
171 int timerfd_init()
172 {
173     int tmfd;
174     int ret;
175     struct itimerspec new_value;
176 
177     new_value.it_value.tv_sec = 2;
178     new_value.it_value.tv_nsec = 0;
179     new_value.it_interval.tv_sec = 1;
180     new_value.it_interval.tv_nsec = 0;
181     
182     tmfd = timerfd_create(CLOCK_MONOTONIC, 0);
183     if (tmfd < 0) {
184         LOG_ERROR("timerfd_create error, Error:[%d:%s]", errno, strerror(errno));
185         return -1;
186     }
187 
188     ret = timerfd_settime(tmfd, 0, &new_value, NULL);
189     if (ret < 0) {
190         LOG_ERROR("timerfd_settime error, Error:[%d:%s]", errno, strerror(errno));
191         close(tmfd);
192         return -1;
193     }
194 
195     if (epoll_add_fd(tmfd)) {
196         close(tmfd);
197         return -1;
198     }
199     g_timerfd = tmfd;
200 
201     return 0;
202 }
203 
204 int main(int argc, char **argv)
205 {
206     if (epollfd_init() < 0) {
207         return -1;
208     }
209 
210     if (timerfd_init()) {
211         return -1;
212     }
213 
214     /* event handle */
215     epoll_event_handle();
216 
217     return 0;
218 }

參考資料:code

http://www.man7.org/linux/man-pages/man2/timerfd_create.2.htmlhtm

http://blog.csdn.net/walkingman321/article/details/6162055對象

相關文章
相關標籤/搜索