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(&params);
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