aboutsummaryrefslogtreecommitdiffstats
path: root/test/test_vevent.c
blob: bfd0369ba53c9ef4a41bd4ab240e798022b03f2a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/*
 * test_vevent.c - Test VIDEO_GET_EVENT and poll(9 for video events
 *
 * Copyright (C) 2003 convergence GmbH
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public License
 * as published by the Free Software Foundation; either version 2.1
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 */

#include <sys/ioctl.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/poll.h>
#include <fcntl.h>
#include <time.h>
#include <unistd.h>

#include <linux/types.h>
#include <linux/dvb/video.h>


int videoGetSize(int fd, char *arg)
{
	video_size_t size;

	if (arg)
		return -1;
	if (ioctl(fd, VIDEO_GET_SIZE, &size) == -1){
		perror("VIDEO_GET_SIZE");
		return 0;
	}

	printf("Video Size: %ux%u ", size.w, size.h);
	switch (size.aspect_ratio) {
	case VIDEO_FORMAT_4_3:
		printf("4:3 (%d)\n", size.aspect_ratio);
		break;
	case VIDEO_FORMAT_16_9:
		printf("16:9 (%d)\n", size.aspect_ratio);
		break;
	case VIDEO_FORMAT_221_1:
		printf("2.21:1 (%d)\n", size.aspect_ratio);
		break;
	default:
		printf("unknown aspect ratio (%d)\n", size.aspect_ratio);
		break;
	}
	return 0;
}

int main(void)
{
	int vfd, rc;
	char *videodev = "/dev/dvb/adapter0/video0";
	struct pollfd pfd[1];
	struct video_event event;

	if (getenv("VIDEO"))
		videodev = getenv("VIDEO");

	printf("using video device '%s'\n", videodev);

	if ((vfd = open(videodev, O_RDONLY | O_NONBLOCK)) < 0) {
		perror("open video device");
		return 1;
	}

	videoGetSize(vfd, NULL);

	pfd[0].fd = vfd;
	pfd[0].events = POLLPRI;

	for (;;) {
		rc = poll(pfd, 1, -1);
		if (rc == -1) {
			perror("poll");
			return -1;
		}
		printf("poll events: %#x\n", pfd[0].revents);
		if (pfd[0].revents & POLLPRI) {
			rc = ioctl(vfd, VIDEO_GET_EVENT, &event);
			if (rc == -1) {
				perror("VIDEO_GET_EVENT");
				return -1;
			}
			printf("video event %d\n", event.type);
			if (event.type == VIDEO_EVENT_SIZE_CHANGED) {
				printf("  VIDEO_EVENT_SIZE_CHANGED %ux%u ",
						event.u.size.w, event.u.size.h);
				switch (event.u.size.aspect_ratio) {
				case VIDEO_FORMAT_4_3:
					printf("4:3 (%d)\n", event.u.size.aspect_ratio);
					break;
				case VIDEO_FORMAT_16_9:
					printf("16:9 (%d)\n", event.u.size.aspect_ratio);
					break;
				case VIDEO_FORMAT_221_1:
					printf("2.21:1 (%d)\n", event.u.size.aspect_ratio);
					break;
				default:
					printf("unknown aspect ratio (%d)\n",
							event.u.size.aspect_ratio);
					break;
				}
			}
		}
	}

	close(vfd);
	return 0;
}
fier_descriptor_entries_for_each(d, pos) \ for ((pos) = dvb_content_identifier_descriptor_entries_first(d); \ (pos); \ (pos) = dvb_content_identifier_descriptor_entries_next(d, pos)) /** * Accessor for the data0 field of a dvb_content_identifier_entry. * * @param d dvb_content_identifier_entry pointer. * @return Pointer, or NULL on error. */ static inline struct dvb_content_identifier_entry_data_0* dvb_content_identifier_entry_data_0(struct dvb_content_identifier_entry *d) { if (d->crid_location != 0) return NULL; return (struct dvb_content_identifier_entry_data_0*) ((uint8_t*) d + sizeof(struct dvb_content_identifier_entry)); } /** * Accessor for the data field of a dvb_content_identifier_entry_data_0. * * @param d dvb_content_identifier_entry_data_0 pointer. * @return Pointer, or NULL on error. */ static inline uint8_t* dvb_content_identifier_entry_data_0_data(struct dvb_content_identifier_entry_data_0 *d) { return ((uint8_t*) d + sizeof(struct dvb_content_identifier_entry_data_0)); } /** * Accessor for the data1 field of a dvb_content_identifier_entry. * * @param d dvb_content_identifier_entry pointer. * @return Pointer, or NULL on error. */ static inline struct dvb_content_identifier_entry_data_1* dvb_content_identifier_entry_data_1(struct dvb_content_identifier_entry *d) { if (d->crid_location != 1) return NULL; return (struct dvb_content_identifier_entry_data_1*) ((uint8_t*) d + sizeof(struct dvb_content_identifier_entry)); } /******************************** PRIVATE CODE ********************************/ static inline struct dvb_content_identifier_entry* dvb_content_identifier_descriptor_entries_first(struct dvb_content_identifier_descriptor *d) { if (d->d.len == 0) return NULL; return (struct dvb_content_identifier_entry *) ((uint8_t*) d + sizeof(struct dvb_content_identifier_descriptor)); } static inline struct dvb_content_identifier_entry* dvb_content_identifier_descriptor_entries_next(struct dvb_content_identifier_descriptor *d, struct dvb_content_identifier_entry *pos) { uint8_t *end = (uint8_t*) d + 2 + d->d.len; uint8_t *next = (uint8_t *) pos + sizeof(struct dvb_content_identifier_entry); if (next >= end) return NULL; switch(pos->crid_location) { case 0: if ((next+2) >= end) return NULL; if ((next+2+next[1]) >= end) return NULL; break; case 1: if ((next+3) >= end) return NULL; break; } return (struct dvb_content_identifier_entry*) next; } #ifdef __cplusplus } #endif #endif