Back to Cpython

timer file descriptor HOWTO

Doc/howto/timerfd.rst

3.15.0a88.5 KB
Original Source

.. _timerfd-howto:


timer file descriptor HOWTO


:Release: 1.13

This HOWTO discusses Python's support for the linux timer file descriptor.

Examples

The following example shows how to use a timer file descriptor to execute a function twice a second:

.. code-block:: python

Practical scripts should use really use a non-blocking timer,

we use a blocking timer here for simplicity.

import os, time

Create the timer file descriptor

fd = os.timerfd_create(time.CLOCK_REALTIME)

Start the timer in 1 second, with an interval of half a second

os.timerfd_settime(fd, initial=1, interval=0.5)

try: # Process timer events four times. for _ in range(4): # read() will block until the timer expires _ = os.read(fd, 8) print("Timer expired") finally: # Remember to close the timer file descriptor! os.close(fd)

To avoid the precision loss caused by the :class:float type, timer file descriptors allow specifying initial expiration and interval in integer nanoseconds with _ns variants of the functions.

This example shows how :func:~select.epoll can be used with timer file descriptors to wait until the file descriptor is ready for reading:

.. code-block:: python

import os, time, select, socket, sys

Create an epoll object

ep = select.epoll()

In this example, use loopback address to send "stop" command to the server.

$ telnet 127.0.0.1 1234

Trying 127.0.0.1...

Connected to 127.0.0.1.

Escape character is '^]'.

stop

Connection closed by foreign host.

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.bind(("127.0.0.1", 1234)) sock.setblocking(False) sock.listen(1) ep.register(sock, select.EPOLLIN)

Create timer file descriptors in non-blocking mode.

num = 3 fds = [] for _ in range(num): fd = os.timerfd_create(time.CLOCK_REALTIME, flags=os.TFD_NONBLOCK) fds.append(fd) # Register the timer file descriptor for read events ep.register(fd, select.EPOLLIN)

Start the timer with os.timerfd_settime_ns() in nanoseconds.

Timer 1 fires every 0.25 seconds; timer 2 every 0.5 seconds; etc

for i, fd in enumerate(fds, start=1): one_sec_in_nsec = 10**9 i = i * one_sec_in_nsec os.timerfd_settime_ns(fd, initial=i//4, interval=i//4)

timeout = 3 try: conn = None is_active = True while is_active: # Wait for the timer to expire for 3 seconds. # epoll.poll() returns a list of (fd, event) pairs. # fd is a file descriptor. # sock and conn[=returned value of socket.accept()] are socket objects, not file descriptors. # So use sock.fileno() and conn.fileno() to get the file descriptors. events = ep.poll(timeout)

       # If more than one timer file descriptors are ready for reading at once,
       # epoll.poll() returns a list of (fd, event) pairs.
       #
       # In this example settings,
       #    1st timer fires every 0.25 seconds in 0.25 seconds. (0.25, 0.5, 0.75, 1.0, ...)
       #    2nd timer every 0.5 seconds in 0.5 seconds. (0.5, 1.0, 1.5, 2.0, ...)
       #    3rd timer every 0.75 seconds in 0.75 seconds. (0.75, 1.5, 2.25, 3.0, ...)
       #
       #    In 0.25 seconds, only 1st timer fires.
       #    In 0.5 seconds, 1st timer and 2nd timer fires at once.
       #    In 0.75 seconds, 1st timer and 3rd timer fires at once.
       #    In 1.5 seconds, 1st timer, 2nd timer and 3rd timer fires at once.
       #
       # If a timer file descriptor is signaled more than once since
       # the last os.read() call, os.read() returns the number of signaled
       # as host order of class bytes.
       print(f"Signaled events={events}")
       for fd, event in events:
           if event & select.EPOLLIN:
               if fd == sock.fileno():
                   # Check if there is a connection request.
                   print(f"Accepting connection {fd}")
                   conn, addr = sock.accept()
                   conn.setblocking(False)
                   print(f"Accepted connection {conn} from {addr}")
                   ep.register(conn, select.EPOLLIN)
               elif conn and fd == conn.fileno():
                   # Check if there is data to read.
                   print(f"Reading data {fd}")
                   data = conn.recv(1024)
                   if data:
                       # You should catch UnicodeDecodeError exception for safety.
                       cmd = data.decode()
                       if cmd.startswith("stop"):
                           print(f"Stopping server")
                           is_active = False
                       else:
                           print(f"Unknown command: {cmd}")
                   else:
                       # No more data, close connection
                       print(f"Closing connection {fd}")
                       ep.unregister(conn)
                       conn.close()
                       conn = None
               elif fd in fds:
                   print(f"Reading timer {fd}")
                   count = int.from_bytes(os.read(fd, 8), byteorder=sys.byteorder)
                   print(f"Timer {fds.index(fd) + 1} expired {count} times")
               else:
                   print(f"Unknown file descriptor {fd}")

finally: for fd in fds: ep.unregister(fd) os.close(fd) ep.close()

This example shows how :func:~select.select can be used with timer file descriptors to wait until the file descriptor is ready for reading:

.. code-block:: python

import os, time, select, socket, sys

In this example, use loopback address to send "stop" command to the server.

$ telnet 127.0.0.1 1234

Trying 127.0.0.1...

Connected to 127.0.0.1.

Escape character is '^]'.

stop

Connection closed by foreign host.

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.bind(("127.0.0.1", 1234)) sock.setblocking(False) sock.listen(1)

Create timer file descriptors in non-blocking mode.

num = 3 fds = [os.timerfd_create(time.CLOCK_REALTIME, flags=os.TFD_NONBLOCK) for _ in range(num)] select_fds = fds + [sock]

Start the timers with os.timerfd_settime() in seconds.

Timer 1 fires every 0.25 seconds; timer 2 every 0.5 seconds; etc

for i, fd in enumerate(fds, start=1): os.timerfd_settime(fd, initial=i/4, interval=i/4)

timeout = 3 try: conn = None is_active = True while is_active: # Wait for the timer to expire for 3 seconds. # select.select() returns a list of file descriptors or objects. rfd, wfd, xfd = select.select(select_fds, select_fds, select_fds, timeout) for fd in rfd: if fd == sock: # Check if there is a connection request. print(f"Accepting connection {fd}") conn, addr = sock.accept() conn.setblocking(False) print(f"Accepted connection {conn} from {addr}") select_fds.append(conn) elif conn and fd == conn: # Check if there is data to read. print(f"Reading data {fd}") data = conn.recv(1024) if data: # You should catch UnicodeDecodeError exception for safety. cmd = data.decode() if cmd.startswith("stop"): print(f"Stopping server") is_active = False else: print(f"Unknown command: {cmd}") else: # No more data, close connection print(f"Closing connection {fd}") select_fds.remove(conn) conn.close() conn = None elif fd in fds: print(f"Reading timer {fd}") count = int.from_bytes(os.read(fd, 8), byteorder=sys.byteorder) print(f"Timer {fds.index(fd) + 1} expired {count} times") else: print(f"Unknown file descriptor {fd}") finally: for fd in fds: os.close(fd) sock.close() sock = None