1d5ac70f0Sopenharmony_ci#include "config.h" 2d5ac70f0Sopenharmony_ci#include <stdio.h> 3d5ac70f0Sopenharmony_ci#include <stdlib.h> 4d5ac70f0Sopenharmony_ci#include <string.h> 5d5ac70f0Sopenharmony_ci#include <sys/time.h> 6d5ac70f0Sopenharmony_ci#include "../include/asoundlib.h" 7d5ac70f0Sopenharmony_ci 8d5ac70f0Sopenharmony_civoid show_status(void *handle) 9d5ac70f0Sopenharmony_ci{ 10d5ac70f0Sopenharmony_ci int err; 11d5ac70f0Sopenharmony_ci snd_timer_status_t *status; 12d5ac70f0Sopenharmony_ci 13d5ac70f0Sopenharmony_ci snd_timer_status_alloca(&status); 14d5ac70f0Sopenharmony_ci if ((err = snd_timer_status(handle, status)) < 0) { 15d5ac70f0Sopenharmony_ci fprintf(stderr, "timer status %i (%s)\n", err, snd_strerror(err)); 16d5ac70f0Sopenharmony_ci return; 17d5ac70f0Sopenharmony_ci } 18d5ac70f0Sopenharmony_ci printf("STATUS:\n"); 19d5ac70f0Sopenharmony_ci printf(" resolution = %li\n", snd_timer_status_get_resolution(status)); 20d5ac70f0Sopenharmony_ci printf(" lost = %li\n", snd_timer_status_get_lost(status)); 21d5ac70f0Sopenharmony_ci printf(" overrun = %li\n", snd_timer_status_get_overrun(status)); 22d5ac70f0Sopenharmony_ci printf(" queue = %li\n", snd_timer_status_get_queue(status)); 23d5ac70f0Sopenharmony_ci} 24d5ac70f0Sopenharmony_ci 25d5ac70f0Sopenharmony_civoid read_loop(void *handle, int master_ticks, int timeout) 26d5ac70f0Sopenharmony_ci{ 27d5ac70f0Sopenharmony_ci int count, err; 28d5ac70f0Sopenharmony_ci struct pollfd *fds; 29d5ac70f0Sopenharmony_ci snd_timer_read_t tr; 30d5ac70f0Sopenharmony_ci 31d5ac70f0Sopenharmony_ci count = snd_timer_poll_descriptors_count(handle); 32d5ac70f0Sopenharmony_ci fds = calloc(count, sizeof(struct pollfd)); 33d5ac70f0Sopenharmony_ci if (fds == NULL) { 34d5ac70f0Sopenharmony_ci fprintf(stderr, "malloc error\n"); 35d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 36d5ac70f0Sopenharmony_ci } 37d5ac70f0Sopenharmony_ci while (master_ticks-- > 0) { 38d5ac70f0Sopenharmony_ci if ((err = snd_timer_poll_descriptors(handle, fds, count)) < 0) { 39d5ac70f0Sopenharmony_ci fprintf(stderr, "snd_timer_poll_descriptors error: %s\n", snd_strerror(err)); 40d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 41d5ac70f0Sopenharmony_ci } 42d5ac70f0Sopenharmony_ci if ((err = poll(fds, count, timeout)) < 0) { 43d5ac70f0Sopenharmony_ci fprintf(stderr, "poll error %i (%s)\n", err, strerror(err)); 44d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 45d5ac70f0Sopenharmony_ci } 46d5ac70f0Sopenharmony_ci if (err == 0) { 47d5ac70f0Sopenharmony_ci fprintf(stderr, "timer time out!!\n"); 48d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 49d5ac70f0Sopenharmony_ci } 50d5ac70f0Sopenharmony_ci while (snd_timer_read(handle, &tr, sizeof(tr)) == sizeof(tr)) { 51d5ac70f0Sopenharmony_ci printf("TIMER: resolution = %uns, ticks = %u\n", 52d5ac70f0Sopenharmony_ci tr.resolution, tr.ticks); 53d5ac70f0Sopenharmony_ci } 54d5ac70f0Sopenharmony_ci } 55d5ac70f0Sopenharmony_ci free(fds); 56d5ac70f0Sopenharmony_ci} 57d5ac70f0Sopenharmony_ci 58d5ac70f0Sopenharmony_cistatic void async_callback(snd_async_handler_t *ahandler) 59d5ac70f0Sopenharmony_ci{ 60d5ac70f0Sopenharmony_ci snd_timer_t *handle = snd_async_handler_get_timer(ahandler); 61d5ac70f0Sopenharmony_ci int *acount = snd_async_handler_get_callback_private(ahandler); 62d5ac70f0Sopenharmony_ci snd_timer_read_t tr; 63d5ac70f0Sopenharmony_ci 64d5ac70f0Sopenharmony_ci while (snd_timer_read(handle, &tr, sizeof(tr)) == sizeof(tr)) { 65d5ac70f0Sopenharmony_ci printf("TIMER: resolution = %uns, ticks = %u\n", 66d5ac70f0Sopenharmony_ci tr.resolution, tr.ticks); 67d5ac70f0Sopenharmony_ci } 68d5ac70f0Sopenharmony_ci (*acount)++; 69d5ac70f0Sopenharmony_ci} 70d5ac70f0Sopenharmony_ci 71d5ac70f0Sopenharmony_ciint main(int argc, char *argv[]) 72d5ac70f0Sopenharmony_ci{ 73d5ac70f0Sopenharmony_ci int idx, err; 74d5ac70f0Sopenharmony_ci int class = SND_TIMER_CLASS_GLOBAL; 75d5ac70f0Sopenharmony_ci int sclass = SND_TIMER_CLASS_NONE; 76d5ac70f0Sopenharmony_ci int card = 0; 77d5ac70f0Sopenharmony_ci int device = SND_TIMER_GLOBAL_SYSTEM; 78d5ac70f0Sopenharmony_ci int subdevice = 0; 79d5ac70f0Sopenharmony_ci int list = 0; 80d5ac70f0Sopenharmony_ci int async = 0; 81d5ac70f0Sopenharmony_ci int acount = 0; 82d5ac70f0Sopenharmony_ci snd_timer_t *handle; 83d5ac70f0Sopenharmony_ci snd_timer_id_t *id; 84d5ac70f0Sopenharmony_ci snd_timer_info_t *info; 85d5ac70f0Sopenharmony_ci snd_timer_params_t *params; 86d5ac70f0Sopenharmony_ci char timername[64]; 87d5ac70f0Sopenharmony_ci snd_async_handler_t *ahandler; 88d5ac70f0Sopenharmony_ci 89d5ac70f0Sopenharmony_ci snd_timer_id_alloca(&id); 90d5ac70f0Sopenharmony_ci snd_timer_info_alloca(&info); 91d5ac70f0Sopenharmony_ci snd_timer_params_alloca(¶ms); 92d5ac70f0Sopenharmony_ci 93d5ac70f0Sopenharmony_ci idx = 1; 94d5ac70f0Sopenharmony_ci while (idx < argc) { 95d5ac70f0Sopenharmony_ci if (!strncmp(argv[idx], "class=", 5)) { 96d5ac70f0Sopenharmony_ci class = atoi(argv[idx]+6); 97d5ac70f0Sopenharmony_ci } else if (!strncmp(argv[idx], "sclass=", 6)) { 98d5ac70f0Sopenharmony_ci sclass = atoi(argv[idx]+7); 99d5ac70f0Sopenharmony_ci } else if (!strncmp(argv[idx], "card=", 5)) { 100d5ac70f0Sopenharmony_ci card = atoi(argv[idx]+5); 101d5ac70f0Sopenharmony_ci } else if (!strncmp(argv[idx], "device=", 7)) { 102d5ac70f0Sopenharmony_ci device = atoi(argv[idx]+7); 103d5ac70f0Sopenharmony_ci } else if (!strncmp(argv[idx], "subdevice=", 10)) { 104d5ac70f0Sopenharmony_ci subdevice = atoi(argv[idx]+10); 105d5ac70f0Sopenharmony_ci } else if (!strcmp(argv[idx], "list")) { 106d5ac70f0Sopenharmony_ci list = 1; 107d5ac70f0Sopenharmony_ci } else if (!strcmp(argv[idx], "async")) { 108d5ac70f0Sopenharmony_ci async = 1; 109d5ac70f0Sopenharmony_ci } 110d5ac70f0Sopenharmony_ci idx++; 111d5ac70f0Sopenharmony_ci } 112d5ac70f0Sopenharmony_ci if (class == SND_TIMER_CLASS_SLAVE && sclass == SND_TIMER_SCLASS_NONE) { 113d5ac70f0Sopenharmony_ci fprintf(stderr, "slave class is not set\n"); 114d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 115d5ac70f0Sopenharmony_ci } 116d5ac70f0Sopenharmony_ci if (list) { 117d5ac70f0Sopenharmony_ci snd_timer_query_t *qhandle; 118d5ac70f0Sopenharmony_ci if ((err = snd_timer_query_open(&qhandle, "hw", 0)) < 0) { 119d5ac70f0Sopenharmony_ci fprintf(stderr, "snd_timer_query_open error: %s\n", snd_strerror(err)); 120d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 121d5ac70f0Sopenharmony_ci } 122d5ac70f0Sopenharmony_ci snd_timer_id_set_class(id, SND_TIMER_CLASS_NONE); 123d5ac70f0Sopenharmony_ci while (1) { 124d5ac70f0Sopenharmony_ci if ((err = snd_timer_query_next_device(qhandle, id)) < 0) { 125d5ac70f0Sopenharmony_ci fprintf(stderr, "timer next device error: %s\n", snd_strerror(err)); 126d5ac70f0Sopenharmony_ci break; 127d5ac70f0Sopenharmony_ci } 128d5ac70f0Sopenharmony_ci if (snd_timer_id_get_class(id) < 0) 129d5ac70f0Sopenharmony_ci break; 130d5ac70f0Sopenharmony_ci printf("Timer device: class %i, sclass %i, card %i, device %i, subdevice %i\n", 131d5ac70f0Sopenharmony_ci snd_timer_id_get_class(id), 132d5ac70f0Sopenharmony_ci snd_timer_id_get_sclass(id), 133d5ac70f0Sopenharmony_ci snd_timer_id_get_card(id), 134d5ac70f0Sopenharmony_ci snd_timer_id_get_device(id), 135d5ac70f0Sopenharmony_ci snd_timer_id_get_subdevice(id)); 136d5ac70f0Sopenharmony_ci } 137d5ac70f0Sopenharmony_ci snd_timer_query_close(qhandle); 138d5ac70f0Sopenharmony_ci exit(EXIT_SUCCESS); 139d5ac70f0Sopenharmony_ci } 140d5ac70f0Sopenharmony_ci sprintf(timername, "hw:CLASS=%i,SCLASS=%i,CARD=%i,DEV=%i,SUBDEV=%i", class, sclass, card, device, subdevice); 141d5ac70f0Sopenharmony_ci if ((err = snd_timer_open(&handle, timername, SND_TIMER_OPEN_NONBLOCK))<0) { 142d5ac70f0Sopenharmony_ci fprintf(stderr, "timer open %i (%s)\n", err, snd_strerror(err)); 143d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 144d5ac70f0Sopenharmony_ci } 145d5ac70f0Sopenharmony_ci printf("Using timer class %i, slave class %i, card %i, device %i, subdevice %i\n", class, sclass, card, device, subdevice); 146d5ac70f0Sopenharmony_ci if ((err = snd_timer_info(handle, info)) < 0) { 147d5ac70f0Sopenharmony_ci fprintf(stderr, "timer info %i (%s)\n", err, snd_strerror(err)); 148d5ac70f0Sopenharmony_ci exit(0); 149d5ac70f0Sopenharmony_ci } 150d5ac70f0Sopenharmony_ci printf("Timer info:\n"); 151d5ac70f0Sopenharmony_ci printf(" slave = %s\n", snd_timer_info_is_slave(info) ? "yes" : "no"); 152d5ac70f0Sopenharmony_ci printf(" card = %i\n", snd_timer_info_get_card(info)); 153d5ac70f0Sopenharmony_ci printf(" id = '%s'\n", snd_timer_info_get_id(info)); 154d5ac70f0Sopenharmony_ci printf(" name = '%s'\n", snd_timer_info_get_name(info)); 155d5ac70f0Sopenharmony_ci printf(" average resolution = %li\n", snd_timer_info_get_resolution(info)); 156d5ac70f0Sopenharmony_ci snd_timer_params_set_auto_start(params, 1); 157d5ac70f0Sopenharmony_ci if (!snd_timer_info_is_slave(info)) { 158d5ac70f0Sopenharmony_ci snd_timer_params_set_ticks(params, (1000000000 / snd_timer_info_get_resolution(info)) / 50); /* 50Hz */ 159d5ac70f0Sopenharmony_ci if (snd_timer_params_get_ticks(params) < 1) 160d5ac70f0Sopenharmony_ci snd_timer_params_set_ticks(params, 1); 161d5ac70f0Sopenharmony_ci printf("Using %li tick(s)\n", snd_timer_params_get_ticks(params)); 162d5ac70f0Sopenharmony_ci } else { 163d5ac70f0Sopenharmony_ci snd_timer_params_set_ticks(params, 1); 164d5ac70f0Sopenharmony_ci } 165d5ac70f0Sopenharmony_ci if ((err = snd_timer_params(handle, params)) < 0) { 166d5ac70f0Sopenharmony_ci fprintf(stderr, "timer params %i (%s)\n", err, snd_strerror(err)); 167d5ac70f0Sopenharmony_ci exit(0); 168d5ac70f0Sopenharmony_ci } 169d5ac70f0Sopenharmony_ci show_status(handle); 170d5ac70f0Sopenharmony_ci if (async) { 171d5ac70f0Sopenharmony_ci err = snd_async_add_timer_handler(&ahandler, handle, async_callback, &acount); 172d5ac70f0Sopenharmony_ci if (err < 0) { 173d5ac70f0Sopenharmony_ci fprintf(stderr, "unable to add async handler %i (%s)\n", err, snd_strerror(err)); 174d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 175d5ac70f0Sopenharmony_ci } 176d5ac70f0Sopenharmony_ci } 177d5ac70f0Sopenharmony_ci if ((err = snd_timer_start(handle)) < 0) { 178d5ac70f0Sopenharmony_ci fprintf(stderr, "timer start %i (%s)\n", err, snd_strerror(err)); 179d5ac70f0Sopenharmony_ci exit(EXIT_FAILURE); 180d5ac70f0Sopenharmony_ci } 181d5ac70f0Sopenharmony_ci if (async) { 182d5ac70f0Sopenharmony_ci /* because all other work is done in the signal handler, 183d5ac70f0Sopenharmony_ci suspend the process */ 184d5ac70f0Sopenharmony_ci while (acount < 25) 185d5ac70f0Sopenharmony_ci sleep(1); 186d5ac70f0Sopenharmony_ci snd_timer_stop(handle); 187d5ac70f0Sopenharmony_ci } else { 188d5ac70f0Sopenharmony_ci read_loop(handle, 25, snd_timer_info_is_slave(info) ? 10000 : 25); 189d5ac70f0Sopenharmony_ci } 190d5ac70f0Sopenharmony_ci show_status(handle); 191d5ac70f0Sopenharmony_ci snd_timer_close(handle); 192d5ac70f0Sopenharmony_ci printf("Done\n"); 193d5ac70f0Sopenharmony_ci return EXIT_SUCCESS; 194d5ac70f0Sopenharmony_ci} 195