/**************************************************************************** * apps/examples/timer/timer_main.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define DEVNAME_SIZE 16 #ifndef CONFIG_EXAMPLES_TIMER_DEVNAME # define CONFIG_EXAMPLES_TIMER_DEVNAME "/dev/timer0" #endif #ifndef CONFIG_EXAMPLES_TIMER_INTERVAL # define CONFIG_EXAMPLES_TIMER_INTERVAL 1000000 #endif #ifndef CONFIG_EXAMPLES_TIMER_DELAY # define CONFIG_EXAMPLES_TIMER_DELAY 100000 #endif #ifndef CONFIG_EXAMPLES_TIMER_NSAMPLES # define CONFIG_EXAMPLES_TIMER_NSAMPLES 20 #endif #ifndef CONFIG_EXAMPLES_TIMER_SIGNO # define CONFIG_EXAMPLES_TIMER_SIGNO 32 #endif /**************************************************************************** * Private Types ****************************************************************************/ /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Private Data ****************************************************************************/ static volatile unsigned long g_nsignals; /**************************************************************************** * timer_sighandler ****************************************************************************/ static void timer_sighandler(int signo, FAR siginfo_t *siginfo, FAR void *context) { /* Does nothing in this example except for increment a count of signals * received. * * NOTE: The use of signal handler is not recommended if you are concerned * about the signal latency. Instead, a dedicated, high-priority thread * that waits on sigwaitinfo() is recommended. High priority is required * if you want a deterministic wake-up time when the signal occurs. */ g_nsignals++; } /**************************************************************************** * timer_status ****************************************************************************/ static void timer_status(int fd) { struct timer_status_s status; int ret; /* Get timer status */ ret = ioctl(fd, TCIOC_GETSTATUS, (unsigned long)((uintptr_t)&status)); if (ret < 0) { fprintf(stderr, "ERROR: Failed to get timer status: %d\n", errno); close(fd); exit(EXIT_FAILURE); } /* Print the timer status */ printf(" flags: %08lx timeout: %lu timeleft: %lu nsignals: %lu\n", (unsigned long)status.flags, (unsigned long)status.timeout, (unsigned long)status.timeleft, g_nsignals); } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * timer_main ****************************************************************************/ int main(int argc, FAR char *argv[]) { struct timer_notify_s notify; struct sigaction act; int ret; int fd; int i; int opt; char devname[DEVNAME_SIZE]; strlcpy(devname, CONFIG_EXAMPLES_TIMER_DEVNAME, sizeof(devname)); while ((opt = getopt(argc, argv, ":d:")) != -1) { switch (opt) { case 'd': strlcpy(devname, optarg, sizeof(devname)); break; case ':': fprintf(stderr, "ERROR: Option needs a value\n"); exit(EXIT_FAILURE); default: /* '?' */ fprintf(stderr, "Usage: %s [-d /dev/timerx]\n", argv[0]); exit(EXIT_FAILURE); } } /* Open the timer device */ printf("Open %s\n", devname); fd = open(devname, O_RDONLY); if (fd < 0) { fprintf(stderr, "ERROR: Failed to open %s: %d\n", devname, errno); return EXIT_FAILURE; } /* Show the timer status before setting the timer interval */ timer_status(fd); /* Set the timer interval */ printf("Set timer interval to %lu\n", (unsigned long)CONFIG_EXAMPLES_TIMER_INTERVAL); ret = ioctl(fd, TCIOC_SETTIMEOUT, CONFIG_EXAMPLES_TIMER_INTERVAL); if (ret < 0) { fprintf(stderr, "ERROR: Failed to set the timer interval: %d\n", errno); close(fd); return EXIT_FAILURE; } /* Show the timer status before attaching the timer handler */ timer_status(fd); /* Attach a signal handler to catch the notifications. NOTE that using * signal handler is very slow. A much more efficient thing to do is to * create a separate pthread that waits on sigwaitinfo() for timer events. * Much less overhead in that case. */ g_nsignals = 0; act.sa_sigaction = timer_sighandler; act.sa_flags = SA_SIGINFO; sigfillset(&act.sa_mask); sigdelset(&act.sa_mask, CONFIG_EXAMPLES_TIMER_SIGNO); ret = sigaction(CONFIG_EXAMPLES_TIMER_SIGNO, &act, NULL); if (ret != OK) { fprintf(stderr, "ERROR: Fsigaction failed: %d\n", errno); close(fd); return EXIT_FAILURE; } /* Register a callback for notifications using the configured signal. * * NOTE: If no callback is attached, the timer stop at the first interrupt. */ printf("Attach timer handler\n"); notify.pid = getpid(); notify.periodic = true; notify.event.sigev_notify = SIGEV_SIGNAL; notify.event.sigev_signo = CONFIG_EXAMPLES_TIMER_SIGNO; notify.event.sigev_value.sival_ptr = NULL; ret = ioctl(fd, TCIOC_NOTIFICATION, (unsigned long)((uintptr_t)¬ify)); if (ret < 0) { fprintf(stderr, "ERROR: Failed to set the timer handler: %d\n", errno); close(fd); return EXIT_FAILURE; } /* Show the timer status before starting */ timer_status(fd); /* Start the timer */ printf("Start the timer\n"); ret = ioctl(fd, TCIOC_START, 0); if (ret < 0) { fprintf(stderr, "ERROR: Failed to start the timer: %d\n", errno); close(fd); return EXIT_FAILURE; } /* Wait a bit showing timer status */ for (i = 0; i < CONFIG_EXAMPLES_TIMER_NSAMPLES; i++) { usleep(CONFIG_EXAMPLES_TIMER_DELAY); timer_status(fd); } /* Stop the timer */ printf("Stop the timer\n"); ret = ioctl(fd, TCIOC_STOP, 0); if (ret < 0) { fprintf(stderr, "ERROR: Failed to stop the timer: %d\n", errno); close(fd); return EXIT_FAILURE; } /* Detach the signal handler */ act.sa_handler = SIG_DFL; sigaction(CONFIG_EXAMPLES_TIMER_SIGNO, &act, NULL); /* Show the timer status before starting */ timer_status(fd); /* Close the timer driver */ printf("Finished\n"); close(fd); return EXIT_SUCCESS; }