mirror of
				https://github.com/alsa-project/alsa-lib.git
				synced 2025-10-29 05:40:25 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			160 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/time.h>
 | |
| #include "../include/asoundlib.h"
 | |
| 
 | |
| void show_status(void *handle)
 | |
| {
 | |
| 	int err;
 | |
| 	snd_timer_status_t *status;
 | |
| 	
 | |
| 	snd_timer_status_alloca(&status);
 | |
| 	if ((err = snd_timer_status(handle, status)) < 0) {
 | |
| 		fprintf(stderr, "timer status %i (%s)\n", err, snd_strerror(err));
 | |
| 		return;
 | |
| 	}
 | |
| 	printf("STATUS:\n");
 | |
| 	printf("  resolution = %li\n", snd_timer_status_get_resolution(status));
 | |
| 	printf("  lost = %li\n", snd_timer_status_get_lost(status));
 | |
| 	printf("  overrun = %li\n", snd_timer_status_get_overrun(status));
 | |
| 	printf("  queue = %li\n", snd_timer_status_get_queue(status));
 | |
| }
 | |
| 
 | |
| void read_loop(void *handle, int master_ticks, int timeout)
 | |
| {
 | |
| 	int count, err;
 | |
| 	struct pollfd *fds;
 | |
| 	snd_timer_read_t tr;
 | |
| 	
 | |
| 	count = snd_timer_poll_descriptors_count(handle);
 | |
| 	fds = calloc(count, sizeof(struct pollfd));
 | |
| 	if (fds == NULL) {
 | |
| 		fprintf(stderr, "malloc error\n");
 | |
| 		exit(EXIT_FAILURE);
 | |
| 	}
 | |
| 	while (master_ticks-- > 0) {
 | |
| 		if ((err = snd_timer_poll_descriptors(handle, fds, count)) < 0) {
 | |
| 			fprintf(stderr, "snd_timer_poll_descriptors error: %s\n", snd_strerror(err));
 | |
| 			exit(EXIT_FAILURE);
 | |
| 		}
 | |
| 		if ((err = poll(fds, count, timeout)) < 0) {
 | |
| 			fprintf(stderr, "poll error %i (%s)\n", err, strerror(err));
 | |
| 			exit(EXIT_FAILURE);
 | |
| 		}
 | |
| 		if (err == 0) {
 | |
| 			fprintf(stderr, "timer time out!!\n");
 | |
| 			exit(EXIT_FAILURE);
 | |
| 		}
 | |
| 		while (snd_timer_read(handle, &tr, sizeof(tr)) == sizeof(tr)) {
 | |
| 			printf("TIMER: resolution = %uns, ticks = %u\n",
 | |
| 				tr.resolution, tr.ticks);
 | |
| 		}
 | |
| 	}
 | |
| 	free(fds);
 | |
| }
 | |
| 
 | |
| int main(int argc, char *argv[])
 | |
| {
 | |
| 	int idx, err;
 | |
| 	int class = SND_TIMER_CLASS_GLOBAL;
 | |
| 	int sclass = SND_TIMER_CLASS_NONE;
 | |
| 	int card = 0;
 | |
| 	int device = SND_TIMER_GLOBAL_SYSTEM;
 | |
| 	int subdevice = 0;
 | |
| 	int list = 0;
 | |
| 	snd_timer_t *handle;
 | |
| 	snd_timer_id_t *id;
 | |
| 	snd_timer_info_t *info;
 | |
| 	snd_timer_params_t *params;
 | |
| 	char timername[64];
 | |
| 
 | |
| 	snd_timer_id_alloca(&id);
 | |
| 	snd_timer_info_alloca(&info);
 | |
| 	snd_timer_params_alloca(¶ms);
 | |
| 
 | |
| 	idx = 1;
 | |
| 	while (idx < argc) {
 | |
| 		if (!strncmp(argv[idx], "class=", 5)) {
 | |
| 			class = atoi(argv[idx]+6);
 | |
| 		} else if (!strncmp(argv[idx], "sclass=", 6)) {
 | |
| 			sclass = atoi(argv[idx]+7);
 | |
| 		} else if (!strncmp(argv[idx], "card=", 5)) {
 | |
| 			card = atoi(argv[idx]+5);
 | |
| 		} else if (!strncmp(argv[idx], "device=", 7)) {
 | |
| 			device = atoi(argv[idx]+7);
 | |
| 		} else if (!strncmp(argv[idx], "subdevice=", 10)) {
 | |
| 			subdevice = atoi(argv[idx]+10);
 | |
| 		} else if (!strcmp(argv[idx], "list")) {
 | |
| 			list = 1;
 | |
| 		}
 | |
| 		idx++;
 | |
| 	}
 | |
| 	if (class == SND_TIMER_CLASS_SLAVE && sclass == SND_TIMER_SCLASS_NONE) {
 | |
| 		fprintf(stderr, "slave class is not set\n");
 | |
| 		exit(EXIT_FAILURE);
 | |
| 	}
 | |
| 	if (list) {
 | |
| 		snd_timer_query_t *qhandle;
 | |
| 		if ((err = snd_timer_query_open(&qhandle, "hw", 0)) < 0) {
 | |
| 			fprintf(stderr, "snd_timer_query_open error: %s\n", snd_strerror(err));
 | |
| 			exit(EXIT_FAILURE);
 | |
| 		}
 | |
| 		snd_timer_id_set_class(id, SND_TIMER_CLASS_NONE);
 | |
| 		while (1) {
 | |
| 			if ((err = snd_timer_query_next_device(qhandle, id)) < 0) {
 | |
| 				fprintf(stderr, "timer next device error: %s\n", snd_strerror(err));
 | |
| 				break;
 | |
| 			}
 | |
| 			if (snd_timer_id_get_class(id) < 0)
 | |
| 				break;
 | |
| 			printf("Timer device: class %i, sclass %i, card %i, device %i, subdevice %i\n",
 | |
| 					snd_timer_id_get_class(id),
 | |
| 					snd_timer_id_get_sclass(id),
 | |
| 					snd_timer_id_get_card(id),
 | |
| 					snd_timer_id_get_device(id),
 | |
| 					snd_timer_id_get_subdevice(id));
 | |
| 		}
 | |
| 		snd_timer_query_close(qhandle);
 | |
| 		exit(EXIT_SUCCESS);
 | |
| 	}
 | |
| 	sprintf(timername, "hw:CLASS=%i,SCLASS=%i,CARD=%i,DEV=%i,SUBDEV=%i", class, sclass, card, device, subdevice);
 | |
| 	if ((err = snd_timer_open(&handle, timername, SND_TIMER_OPEN_NONBLOCK))<0) {
 | |
| 		fprintf(stderr, "timer open %i (%s)\n", err, snd_strerror(err));
 | |
| 		exit(EXIT_FAILURE);
 | |
| 	}
 | |
| 	printf("Using timer class %i, slave class %i, card %i, device %i, subdevice %i\n", class, sclass, card, device, subdevice);
 | |
| 	if ((err = snd_timer_info(handle, info)) < 0) {
 | |
| 		fprintf(stderr, "timer info %i (%s)\n", err, snd_strerror(err));
 | |
| 		exit(0);
 | |
| 	}
 | |
| 	printf("Timer info:\n");
 | |
| 	printf("  slave = %s\n", snd_timer_info_is_slave(info) ? "yes" : "no");
 | |
| 	printf("  card = %i\n", snd_timer_info_get_card(info));
 | |
| 	printf("  id = '%s'\n", snd_timer_info_get_id(info));
 | |
| 	printf("  name = '%s'\n", snd_timer_info_get_name(info));
 | |
| 	printf("  average resolution = %li\n", snd_timer_info_get_resolution(info));
 | |
| 	snd_timer_params_set_auto_start(params, 1);
 | |
| 	if (!snd_timer_info_is_slave(info)) {
 | |
| 		snd_timer_params_set_ticks(params, (1000000000 / snd_timer_info_get_resolution(info)) / 50); /* 50Hz */
 | |
| 		if (snd_timer_params_get_ticks(params) < 1)
 | |
| 			snd_timer_params_set_ticks(params, 1);
 | |
| 		printf("Using %li tick(s)\n", snd_timer_params_get_ticks(params));
 | |
| 	} else {
 | |
| 		snd_timer_params_set_ticks(params, 1);
 | |
| 	}
 | |
| 	if ((err = snd_timer_params(handle, params)) < 0) {
 | |
| 		fprintf(stderr, "timer params %i (%s)\n", err, snd_strerror(err));
 | |
| 		exit(0);
 | |
| 	}
 | |
| 	show_status(handle);
 | |
| 	if ((err = snd_timer_start(handle)) < 0) {
 | |
| 		fprintf(stderr, "timer start %i (%s)\n", err, snd_strerror(err));
 | |
| 		exit(EXIT_FAILURE);
 | |
| 	}
 | |
| 	read_loop(handle, 25, snd_timer_info_is_slave(info) ? 10000 : 25);
 | |
| 	show_status(handle);
 | |
| 	snd_timer_close(handle);
 | |
| 	printf("Done\n");
 | |
| 	return EXIT_SUCCESS;
 | |
| }
 | 
