diff options
Diffstat (limited to 'test')
44 files changed, 7893 insertions, 420 deletions
| diff --git a/test/Makefile b/test/Makefile index ddff9ab..30cb9e3 100644 --- a/test/Makefile +++ b/test/Makefile @@ -1,37 +1,41 @@ -# Makefile for Linux DVB API Version 3 test programs +# Makefile for linuxtv.org dvb-apps/test -CC = gcc -CFLAGS = -g -O2 -W -Wall -I../include +objects  = hex_dump.o lnb.o -TARGETS = \ -	diseqc		\ -	set22k		\ -	sendburst	\ -	setvoltage	\ -	setpid		\ -	video		\ -	test_sections	\ -	test_sec_ne	\ -	test_pes	\ -	test_dvr	\ -	test_dvr_play	\ -	test_tt		\ -	test_av		\ -	test_av_play	\ -	test_vevent	\ -	test_stc	\ -	test_stillimage +binaries = diseqc          \ +           sendburst       \ +           set22k          \ +           setpid          \ +           setvoltage      \ +           test_av         \ +           test_av_play    \ +           test_dvr        \ +           test_dvr_play   \ +           test_pes        \ +           test_sec_ne     \ +           test_sections   \ +           test_stc        \ +           test_stillimage \ +           test_tt         \ +           test_vevent     \ +	   evtest	   \ +	   video	   \ +	   szap2 -#	test		\ -#	test_audio	\ -#	test_front	\ -#	test_switch	\ -#	test_video	\ +.PHONY: all -all: $(TARGETS) +all: $(binaries) +	make -C libdvbcfg $@ +	make -C libdvben50221 $@ +	make -C libesg $@ +	make -C libucsi $@ -test_sections test_sec_ne test_pes test_tt: hex_dump.o +$(binaries): $(objects) -clean: -	rm -f $(TARGETS) *.o +clean:: +	make -C libdvbcfg $@ +	make -C libdvben50221 $@ +	make -C libesg $@ +	make -C libucsi $@ +include ../Make.rules diff --git a/test/README b/test/README index b3f0cac..2b39936 100644 --- a/test/README +++ b/test/README @@ -1,4 +1,4 @@ -Various small test/sample programs for the Linux DVB API Version 2 +Various small test/sample programs for the Linux DVB API Version 2/3  The default devices used by the test programs are generally  /dev/dvb/adapter0/*0, and can be overridden using environment @@ -33,13 +33,13 @@ test_vevent	: Test VIDEO_GET_EVENT and poll() for video events  test_stc	: Test DMX_GET_STC.  test_stillimage : Display single iframes as stillimages -		  iframes can be created with the 'convert' tool from  -		  imagemagick and mpeg2encode from ftp.mpeg.org, and must  +		  iframes can be created with the 'convert' tool from +		  imagemagick and mpeg2encode from ftp.mpeg.org, and must  		  have a supported size, e.g. 702x576  		  ($ convert -sample 702x576\! test.jpg test.mpg) -(test_av_play	: Test playing MPEG TS from a file (apparently broken)) - +test_av_play	: Test playing MPEG PES (VDR format) from a file +test_dvr_play	: Test playing MPEG TS from a file (don't try, driver is broken)  test		:  test_audio	: @@ -48,4 +48,3 @@ test_dvr	:  test_front	:  test_switch	:  test_video	: - @@ -4,4 +4,4 @@ for f in $@  do /usr/X11R6/bin/convert  -geomtry 702x576 $f test.mpg  test_video test.mpg  rm test.mpg -done
\ No newline at end of file +done diff --git a/test/diseqc.c b/test/diseqc.c index 8f2e411..5deb7fe 100644 --- a/test/diseqc.c +++ b/test/diseqc.c @@ -1,11 +1,13 @@ -/* - * Test sending DiSEqC commands on a SAT frontend. - * - * usage: FRONTEND=/dev/dvb/adapterX/frontendX diseqc [test_seq_no] - */ +#define USAGE \ +"\n" \ +"\nTest sending DiSEqC commands on a SAT frontend." \ +"\n" \ +"\nusage: FRONTEND=/dev/dvb/adapterX/frontendX diseqc [test_seq_no|'all']" \ +"\n"  #include <pthread.h>  #include <time.h> +#include <string.h>  #include <stdlib.h>  #include <stdint.h>  #include <stdio.h> @@ -109,7 +111,10 @@ int main(int argc, char **argv)  		return -1;  	} -	if (argc > 1) { +	if (argc != 2) { +	    fprintf (stderr, "usage: %s [number|'all']\n" USAGE, argv[0]); +	    return 1; +	} else if (strcmp(argv[1], "all")) {  		int i = atol(argv[1]);  		cmd[0] = &switch_cmds[i];  		diseqc_send_msg(fd, @@ -136,5 +141,3 @@ int main(int argc, char **argv)  	return 0;  } - - diff --git a/test/evtest.c b/test/evtest.c new file mode 100644 index 0000000..a61593e --- /dev/null +++ b/test/evtest.c @@ -0,0 +1,251 @@ +/* + * $Id: evtest.c,v 1.3 2005/08/15 20:43:52 js Exp $ + * + *  Copyright (c) 1999-2000 Vojtech Pavlik + * + *  Event device test program + */ + +/* + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 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 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 + * + * Should you need to contact me, the author, you can do so either by + * e-mail - mail your message to <vojtech@ucw.cz>, or by paper mail: + * Vojtech Pavlik, Simunkova 1594, Prague 8, 182 00 Czech Republic + */ + +#include <linux/input.h> + +#include <string.h> +#include <fcntl.h> +#include <unistd.h> +#include <stdio.h> +#include <stdlib.h> + +char *events[EV_MAX + 1] = { "Sync", "Key", "Relative", "Absolute", "Misc", NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +NULL, NULL, NULL, "LED", "Sound", NULL, "Repeat", "ForceFeedback", "Power", "ForceFeedbackStatus"}; +char *syncs[2] = { "Sync", "Config"}; +char *keys[KEY_MAX + 1] = { +  // 0x000 +  "Reserved", "Esc", "1", "2", "3", "4", "5", "6", +  "7", "8", "9", "0", "Minus", "Equal", "Backspace", "Tab", +  // 0x010 +  "Q", "W", "E", "R", "T", "Y", "U", "I", +  "O", "P", "LeftBrace", "RightBrace", "Enter", "LeftControl", "A", "S", +  // 0x020 +  "D", "F", "G", "H", "J", "K", "L", "Semicolon", +  "Apostrophe", "Grave", "LeftShift", "BackSlash", "Z", "X", "C", "V", +  // 0x030 +  "B", "N", "M", "Comma", "Dot", "Slash", "RightShift", "KPAsterisk", +  "LeftAlt", "Space", "CapsLock", "F1", "F2", "F3", "F4", "F5", +  // 0x040 +  "F6", "F7", "F8", "F9", "F10", "NumLock", "ScrollLock", "KP7", +  "KP8", "KP9", "KPMinus", "KP4", "KP5", "KP6", "KPPlus", "KP1", +  // 0x050 +  "KP2", "KP3", "KP0", "KPDot", NULL, "ZENKAKUHANKAKU", "102nd", "F11", +  "F12", "RO", "KATAKANA", "HIRAGANA", "HENKAN", "KATAKANAHIRAGANA", "MUHENKAN", "KPJPCOMMA", +  // 0x060 +  "KPEnter", "RightCtrl", "KPSlash", "SysRq", "RightAlt", "LineFeed", "Home", "Up", +  "PageUp", "Left", "Right", "End", "Down", "PageDown", "Insert", "Delete", +  // 0x070 +  "Macro", "Mute", "VolumeDown", "VolumeUp", "Power", "KPEqual", "KPPlusMinus", "Pause", +  NULL, "KPComma", "HANGUEL", "HANJA", "YEN", "LeftMeta", "RightMeta", "Compose", +  // 0x080 +  "Stop", "Again", "Props", "Undo", "Front", "Copy", "Open", "Paste", +  "Find", "Cut", "Help", "Menu", "Calc", "Setup", "Sleep", "WakeUp", +  // 0x090 +  "File", "SendFile", "DeleteFile", "X-fer", "Prog1", "Prog2", "WWW", "MSDOS", +  "Coffee", "Direction", "CycleWindows", "Mail", "Bookmarks", "Computer", "Back", "Forward", +  // 0x0A0 +  "CloseCD", "EjectCD", "EjectCloseCD", "NextSong", "PlayPause", "PreviousSong", "StopCD", "Record", +  "Rewind", "Phone", "ISOKey", "Config", "HomePage", "Refresh", "Exit", "Move", +  // 0x0B0 +  "Edit", "ScrollUp", "ScrollDown", "KPLeftParenthesis", "KPRightParenthesis", NULL, NULL, "F13", +  "F14", "F15", "F16", "F17", "F18", "F19", "F20", "F21", +  // 0x0C0 +  "F22", "F23", "F24", NULL, NULL, NULL, NULL, NULL, +  "PlayCD", "PauseCD", "Prog3", "Prog4", NULL, "Suspend", "Close", "Play", +  // 0x0D0 +  "FastForward", "BassBoost", "Print", "HP", "Camera", "Sound", "Question", "EMail", +  "Chat", "Search", "Connect", "Finance", "Sport", "Shop", "AltErase", "Cancel", +  // 0x0E0 +  "BrightnessDown", "BrightnessUp", "Media", NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x0F0 +  "Unknown", NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x100 +  "Btn0", "Btn1", "Btn2", "Btn3", "Btn4", "Btn5", "Btn6", "Btn7", +  "Btn8", "Btn9", NULL, NULL,  NULL, NULL, NULL, NULL, +  // 0x110 +  "LeftBtn", "RightBtn", "MiddleBtn", "SideBtn", "ExtraBtn", "ForwardBtn", "BackBtn", "TaskBtn", +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x120 +  "Trigger", "ThumbBtn", "ThumbBtn2", "TopBtn", "TopBtn2", "PinkieBtn", "BaseBtn", "BaseBtn2", +  "BaseBtn3", "BaseBtn4", "BaseBtn5", "BaseBtn6", NULL, NULL, NULL, "BtnDead", +  // 0x130 +  "BtnA", "BtnB", "BtnC", "BtnX", "BtnY", "BtnZ", "BtnTL", "BtnTR", +  "BtnTL2", "BtnTR2", "BtnSelect", "BtnStart", "BtnMode", "BtnThumbL", "BtnThumbR", NULL, +  // 0x140 +  "ToolPen", "ToolRubber", "ToolBrush", "ToolPencil", "ToolAirbrush", "ToolFinger", "ToolMouse", "ToolLens", +  NULL, NULL, "Touch", "Stylus", "Stylus2", "ToolDoubleTap", "ToolTripleTap", NULL, +  // 0x150 +  "GearDown", "GearUp", NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x160 +  "Ok", "Select", "Goto", "Clear", "Power2", "Option", "Info", "Time", +  "Vendor", "Archive", "Program", "Channel", "Favorites", "EPG", "PVR", "MHP", +  // 0x170 +  "Language", "Title", "Subtitle", "Angle", "Zoom", "Mode", "Keyboard", "Screen", +  "PC", "TV", "TV2", "VCR", "VCR2", "Sat", "Sat2", "CD", +  // 0x180 +  "Tape", "Radio", "Tuner", "Player", "Text", "DVD", "Aux", "MP3", +  "Audio", "Video", "Directory", "List", "Memo", "Calendar", "Red", "Green", +  // 0x190 +  "Yellow", "Blue", "ChannelUp", "ChannelDown", "First", "Last", "AB", "Next", +  "Restart", "Slow", "Shuffle", "Break", "Previous", "Digits", "Teen", "Twen", +  // 0x1A0 +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x1B0 +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x1C0 +  "DelEOL", "DelEOS", "InsLine", "DelLine", NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x1D0 +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x1E0 +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x1F0 +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +}; + +char *absval[5] = { "Value", "Min  ", "Max  ", "Fuzz ", "Flat " }; +char *relatives[REL_MAX + 1] = { +  "X", "Y", "Z", NULL, NULL, NULL, "HWheel", "Dial", +  "Wheel", "Misc", NULL, NULL, NULL, NULL, NULL, NULL, +}; +char *absolutes[ABS_MAX + 1] = { +  // 0x00 +  "X", "Y", "Z", "Rx", "Ry", "Rz", "Throttle", "Rudder", +  "Wheel", "Gas", "Brake", NULL, NULL, NULL, NULL, NULL, +  // 0x10 +  "Hat0X", "Hat0Y", "Hat1X", "Hat1Y", "Hat2X", "Hat2Y", "Hat3X", "Hat 3Y", +  "Pressure", "Distance", "XTilt", "YTilt", "ToolWidth", NULL, NULL, NULL, +  // 0x20 +  "Volume", NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  "Misc", NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  // 0x30 +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +  NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +}; +char *leds[LED_MAX + 1] = { "NumLock", "CapsLock", "ScrollLock", "Compose", "Kana", "Sleep", "Suspend", "Mute" }; +char *repeats[REP_MAX + 1] = { "Delay", "Period" }; +char *sounds[SND_MAX + 1] = { "Bell", "Click" }; + +char **names[EV_MAX + 1] = { syncs, keys, relatives, absolutes, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, +NULL, NULL, leds, sounds, NULL, repeats, NULL, NULL, NULL }; + +#ifndef BITS_PER_LONG +#define BITS_PER_LONG (sizeof(long) * 8) +#endif +#define NBITS(x) ((((x)-1)/BITS_PER_LONG)+1) +#define OFF(x)  ((x)%BITS_PER_LONG) +#define BIT(x)  (1UL<<OFF(x)) +#define LONG(x) ((x)/BITS_PER_LONG) +#define test_bit(bit, array)	((array[LONG(bit)] >> OFF(bit)) & 1) + +int main (int argc, char **argv) +{ +	int fd, rd, i, j, k; +	struct input_event ev[64]; +	int version; +	unsigned short id[4]; +	unsigned long bit[EV_MAX][NBITS(KEY_MAX)]; +	char name[256] = "Unknown"; +	int _abs[5]; + +	if (argc < 2) { +		printf("Usage: evtest /dev/input/eventX\n"); +		printf("Where X = input device number\n"); +		exit(1); +	} + +	if ((fd = open(argv[argc - 1], O_RDONLY)) < 0) { +		perror("evtest"); +		exit(1); +	} + +	if (ioctl(fd, EVIOCGVERSION, &version)) { +		perror("evtest: can't get version"); +		exit(1); +	} + +	printf("Input driver version is %d.%d.%d\n", +		version >> 16, (version >> 8) & 0xff, version & 0xff); + +	ioctl(fd, EVIOCGID, id); +	printf("Input device ID: bus 0x%x vendor 0x%x product 0x%x version 0x%x\n", +		id[ID_BUS], id[ID_VENDOR], id[ID_PRODUCT], id[ID_VERSION]); + +	ioctl(fd, EVIOCGNAME(sizeof(name)), name); +	printf("Input device name: \"%s\"\n", name); + +	memset(bit, 0, sizeof(bit)); +	ioctl(fd, EVIOCGBIT(0, EV_MAX), bit[0]); +	printf("Supported events:\n"); + +	for (i = 0; i < EV_MAX; i++) +		if (test_bit(i, bit[0])) { +			printf("  Event type %d (%s)\n", i, events[i] ? events[i] : "?"); +			ioctl(fd, EVIOCGBIT(i, KEY_MAX), bit[i]); +			for (j = 0; j < KEY_MAX; j++) +				if (test_bit(j, bit[i])) { +					printf("    Event code %d (%s)\n", j, names[i] ? (names[i][j] ? names[i][j] : "?") : "?"); +					if (i == EV_ABS) { +						ioctl(fd, EVIOCGABS(j), _abs); +						for (k = 0; k < 5; k++) +							if ((k < 3) || _abs[k]) +								printf("      %s %6d\n", absval[k], _abs[k]); +					} +				} +		} + + +	printf("Testing ... (interrupt to exit)\n"); + +	while (1) { +		rd = read(fd, ev, sizeof(struct input_event) * 64); + +		if (rd < (int) sizeof(struct input_event)) { +			printf("yyy\n"); +			perror("\nevtest: error reading"); +			exit (1); +		} + +		for (i = 0; i < rd / (int) sizeof(struct input_event); i++) +			printf("Event: time %ld.%06ld, type %d (%s), code %d (%s), value %d\n", +				ev[i].time.tv_sec, ev[i].time.tv_usec, ev[i].type, +				events[ev[i].type] ? events[ev[i].type] : "?", +				ev[i].code, +				names[ev[i].type] ? (names[ev[i].type][ev[i].code] ? names[ev[i].type][ev[i].code] : "?") : "?", +				ev[i].value); + +	} +} diff --git a/test/hex_dump.c b/test/hex_dump.c index c7c8ede..b5c08cb 100644 --- a/test/hex_dump.c +++ b/test/hex_dump.c @@ -60,4 +60,3 @@ void hex_dump(uint8_t data[], int bytes)  	}  	printf("\n");  } - diff --git a/test/libdvbcfg/Makefile b/test/libdvbcfg/Makefile new file mode 100644 index 0000000..7705f9b --- /dev/null +++ b/test/libdvbcfg/Makefile @@ -0,0 +1,12 @@ +# Makefile for linuxtv.org dvb-apps/test/libdvbcfg + +binaries = dvbcfg_test + +CPPFLAGS += -I../../lib +LDLIBS   += ../../lib/libdvbcfg/libdvbcfg.a + +.PHONY: all + +all: $(binaries) + +include ../../Make.rules diff --git a/test/libdvbcfg/dvbcfg_test.c b/test/libdvbcfg/dvbcfg_test.c new file mode 100644 index 0000000..de10e43 --- /dev/null +++ b/test/libdvbcfg/dvbcfg_test.c @@ -0,0 +1,100 @@ +/** + * dvbcfg testing. + * + * Copyright (c) 2005 by Andrew de Quincey <adq_dvb@lidskialf.net> + * + * This library 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA + */ + +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <libdvbcfg/dvbcfg_zapchannel.h> + +void syntax(void); + +struct dvbcfg_zapchannel *channels = NULL; +int zapcount = 0; +int zappos = 0; + +int zapload_callback(struct dvbcfg_zapchannel *channel, void *private); +int zapsave_callback(struct dvbcfg_zapchannel *channel, void *private); + +int main(int argc, char *argv[]) +{ +        if (argc != 4) { +                syntax(); +        } + +	if (!strcmp(argv[1], "-zapchannel")) { + +		FILE *f = fopen(argv[2], "r"); +		if (!f) { +			fprintf(stderr, "Unable to load %s\n", argv[2]); +			exit(1); +		} +		dvbcfg_zapchannel_parse(f, zapload_callback, NULL); +		fclose(f); + +		f = fopen(argv[3], "w"); +		if (!f) { +			fprintf(stderr, "Unable to write %s\n", argv[3]); +			exit(1); +		} +		dvbcfg_zapchannel_save(f, zapsave_callback, NULL); +		fclose(f); + +	} else { +                syntax(); +        } + +        exit(0); +} + +int zapload_callback(struct dvbcfg_zapchannel *channel, void *private) +{ +	(void) private; + +	struct dvbcfg_zapchannel *tmp = realloc(channels, (zapcount+1) * sizeof(struct dvbcfg_zapchannel)); +	if (tmp == NULL) { +		fprintf(stderr, "Out of memory\n"); +		exit(1); +	} +	channels = tmp; + +	memcpy(&channels[zapcount++], channel, sizeof(struct dvbcfg_zapchannel)); + +	return 0; +} + +int zapsave_callback(struct dvbcfg_zapchannel *channel, void *private) +{ +	(void) private; + +	if (zappos >= zapcount) +		return 1; + +	memcpy(channel, channels + zappos, sizeof(struct dvbcfg_zapchannel)); +	zappos++; + +	return 0; +} + +void syntax() +{ +        fprintf(stderr, +                "Syntax: dvbcfg_test <-zapchannel> <input filename> <output filename>\n"); +        exit(1); +} diff --git a/test/libdvbcfg/test_zapchannels.txt b/test/libdvbcfg/test_zapchannels.txt new file mode 100644 index 0000000..98a5b41 --- /dev/null +++ b/test/libdvbcfg/test_zapchannels.txt @@ -0,0 +1,446 @@ +# Most of the major channels in the Raleigh Durham Area.  Frequencies the NTSC center freq. +WRAL:707000000:8VSB:33:36 +WNCN:719000000:8VSB:49:52 +WTVD:701000000:8VSB:49:52 +WRAZ:683000000:8VSB:49:52 +WUNC:743000000:8VSB:49:52 +WRDU:551000000:8VSB:33:36 +WLFL:731000000:8VSB:33:36 +3sat:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:210:220 +ARD-Online-Kanal:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1805 +CNBC:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:510:520 +DLF-Köln:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:810 +DLR-Berlin:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:710 +EinsExtra:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:101:102 +EinsFestival:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:201:202 +EinsMuXx:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:301:302 +EuroNews:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2221:2233 +Eurosport:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:410:420 +Fritz:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:901 +KiKa:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:310:320 +MDR FERNSEHEN:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:401:402 +MDR KULTUR:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:801 +MDR info:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1101 +MHP ARD Online-Kanal:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:102 +NDR Fernsehen:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2401:2402 +ORB-Fernsehen:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:501:502 +RADIOmultikulti:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1301 +Radio 3:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:701 +SFB1:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:601:602 +SWR2:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1401 +SÜDWEST BW:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:801:802 +SÜDWEST RP:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:3101:3102 +WDR 3:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1501 +WDR 5:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1601 +ZDF:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:110:120 +ZDFdigitext:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +ZDFdokukanal:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:660:670 +ZDFinfokanal:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:610:620 +ZDFtheaterkanal:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +JUMP:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1001 +SPUTNIK:426000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:1201 +Österreich 1:394000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:169 +ATV 2:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:631:632 +ATV 2:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:631:632 +Adagio:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +All Jazz:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Avante:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:741:743 +B5 aktuell:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3101 +BBC Prime:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:761:762 +BData3:402000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +BData4:402000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +BData5:402000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +BR-alpha:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:701:702 +Barock Fantasie:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Bayerisches FS:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:201:202 +Bayern 1:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3601 +Bayern 4 Klassik:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3001 +BibelTV:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:731:732 +Bloomberg:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +CLASSICA:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:767:768 +COUNTRY:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:656 +Canal 24 Horas:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:991:992 +Club:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:711:713 +Cristal New Age:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +DANCE:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:304 +DW-tv:610000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:634:632 +Das Erste:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:101:102 +Detskij Mir:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:931:932 +ERT-Sat:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:691:692 +Einstein TV:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:701:702 +Euronews:610000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:597:596 +Eurosport News:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:771:772 +Extreme Sports:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:791:793 +Extreme Sports:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Fashion TV:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Fox Kids:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:671:672 +Fox Kids:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:671:673 +GOLD:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:320 +HITLISTE:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:784 +HR XXL:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3501 +JAZZ:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:640 +Jazz legends:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Kabel Wizard:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Kanal 7:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Kanal 7:610000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:49:52 +Kanal D:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:651:652 +Kanal D:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:651:652 +LATIN:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:368 +Landscape:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Leitseite:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2254:0 +Liberty TV:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:721:723 +MTV Base:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:781:782 +MV-Test:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Modem-Setup:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Movie Sounds:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Musica Antica:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Musica Camerata:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +NDR Info:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3701 +NTV international:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +NTVI:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:611:612 +Nashe Kino:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:621:622 +NordwestRadio:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3801 +OLD GOLD:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:336 +Opernfestival:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +PCNE:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:681:682 +PREMIERE SPORT INTERACTIVE:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +PREMIERE DIREKT 1:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +PREMIERE DIREKT 2:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +PREMIERE DIREKT 3:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +PREMIERE DIREKT 4:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2815:2816 +PREMIERE EROTIK:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1279:1280 +PREMIERE NOSTALGIE:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2559:2560 +PREMIERE SERIE:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1023:1024 +PREMIERE SPORT 1:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:255:258 +PREMIERE SPORT 2:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:3839:3840 +PREMIERE START:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:255:256 +Parlamentsfernsehen:610000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:33:36 +Phoenix:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:901:902 +Portal:402000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +RTP Internacional:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:661:662 +Rai 1:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:951:952 +Rai 2:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:961:962 +Rai 3:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:971:972 +SCHLAGER:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:384 +SR 1:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3901 +SR Fernsehen Suedwest:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:501:502 +Show TV:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:911:912 +Sinfonica:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +TGRT:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:921:922 +TM V1.0:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +TV Polonia:434000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:641:642 +TVEi:121000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:981:982 +TW1:113000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:751:752 +Test-R:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:701:702 +Travel:610000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:595:594 +VCR-Setup:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +VH1 Classic:610000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:604:603 +Videotext:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +WDR FERNSEHEN:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:601:602 +ZEE TV:442000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +arte:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:401:403 +hessen fernsehen:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:301:302 +hr-chronos:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3201 +hr-klassik:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3401 +hr2:410000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:3301 +13 TH STREET:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2303:2304 +ALTERNATIVE ROCK:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:544 +BEATE-UHSE.TV:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1023:1024 +CHILLOUT:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:400 +CLASSIC ROCK:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:352 +DISCOVERY CHANNEL:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1791:1792 +DISNEY CHANNEL:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2559:2560 +DEUTSCHE HITS:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:800 +EASY LISTENING:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:608 +Einstein:346000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +FILM & MUSICAL:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:592 +FOX KIDS:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1279:1280 +GOLDSTAR TV:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:3839:3840 +HARD ROCK:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:816 +HEIMATKANAL:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1535:1536 +HIP HOP/R&B:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:576 +JUNIOR:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:255:256 +K-TOON:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +KLASSIK POPULÄR:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:624 +KRIMI &CO:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1535:1536 +LOVE SONGS:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:592 +NEW COUNTRY:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:384 +ORCHESTRALE WERKE:378000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:560 +PLANET:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1791:1792 +PREMIERE 1:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:511:512 +PREMIERE 2:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1791:1793 +PREMIERE 3:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2303:2304 +PREMIERE 4:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:767:768 +PREMIERE 5:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1279:1280 +PREMIERE 6:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1535:1536 +PREMIERE 7:370000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:1023:1024 +SOUL CLASSICS:362000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:352 +STUDIO UNIVERSAL:354000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:2047:2048 +Sonnenklar TV:402000000:INVERSION_OFF:6900000:FEC_NONE:QAM_64:0:0 +Das Erste:11837:h:0:27500:101:102:28106 +ZDF:11954:h:0:27500:110:120:28006 +3sat:11954:h:0:27500:210:220:28007 +EinsMuXx:12110:h:0:27500:301:302:28203 +EinsFestival:12110:h:0:27500:201:202:28202 +EinsExtra:12110:h:0:27500:101:102:28201 +MDR FERNSEHEN:12110:h:0:27500:401:402:28204 +ORB-Fernsehen:12110:h:0:27500:501:502:28205 +B1 Berlin:12110:h:0:27500:601:602:28206 +SWR Fernsehen:11837:h:0:27500:801:802:28113 +SR Fernsehen Suedwes:11837:h:0:27500:501:502:28110 +hessen fernsehen:11837:h:0:27500:301:302:28108 +WDR FERNSEHEN:11837:h:0:27500:601:602:28111 +Bayerisches FS:11837:h:0:27500:201:202:28107 +N3:12110:h:0:27500:2401:2402:28224 +BR-alpha:11837:h:0:27500:701:702:28112 +KiKa:11954:h:0:27500:310:320:28008 +arte:11836:h:0:27500:401:402:28109 +ZDF Theaterkanal:11954:h:0:27500:1110:1120:28016 +ZDF.info:11954:h:0:27500:610:620:28011 +ZDF.doku:11954:h:0:27500:660:670:28014 +Phoenix:11837:h:0:27500:901:902:28114 +DW-tv:10786:v:0:21997:305:306:9005 +RTL Television:12188:h:0:27500:163:104:12003 +SAT.1:12480:v:0:27500:1791:1792:46 +ProSieben:12480:v:0:27500:255:256:898 +RTL2:12188:h:0:27500:166:128:12020 +Super RTL:12188:h:0:27500:165:120:12040 +KABEL1:12480:v:0:27500:511:512:899 +VOX:12188:h:0:27500:167:136:12060 +tm3:12480:v:0:27500:767:768:897 +Bloomberg TV Germany:12552:v:0:22000:162:99:12160 +EuroNews:11954:h:0:27500:2221:2233:28015 +N24:12480:v:0:27500:2047:2048:47 +n-tv:12670:v:0:22000:162:96:12730 +DSF:12480:v:0:27500:1023:1024:900 +Eurosport:11954:h:0:27500:410:420:28009 +Via 1 - Sch ner Re:12148:h:0:27500:511:512:44 +Home Order Tel:12480:v:0:27500:1279:1280:40 +QVC GERMANY:12552:v:0:22000:165:166:12100 +TW 1:12692:h:0:22000:166:167:13013 +Canal Canarias:12441:v:0:27500:513:681:29700 +ProSieben A:12051:v:0:27500:161:84:20002 +ProSieben CH:12051:v:0:27500:289:290:20001 +Kabel 1 Austria:12051:v:0:27500:166:167:20004 +Kabel 1 Schweiz:12051:v:0:27500:162:163:20003 +CNN Int.:12168:v:0:27500:165:100:28512 +Sky News:12552:v:0:22000:305:306:3995 +Travel:12168:v:0:27500:163:92:28001 +AB SAT / XXL:12266:h:0:27500:164:96:17004 +MOTEURS:12266:h:0:27500:160:80:17000 +HOT GM:12148:h:0:27500:767:768:45 +KTO:12129:v:0:27500:170:120:8411 +LA CINQUIEME:12207:v:0:27500:160:80:8501 +LCP:12207:v:0:27500:165:100:8506 +LibertyTV.com:12611:v:0:22000:941:942:12280 +TV5 Europe:12611:v:0:22000:45:46:12240 +Motors TV:12611:v:0:22000:191:194:12300 +Wishline:12611:v:0:22000:214:216:12320 +TV 5:10786:v:0:21997:164:112:9001 +RTM - MAROC:10786:v:0:21997:162:96:9002 +ESC1 - EGYPTE:10786:v:0:21997:163:104:9003 +RAI 1:10786:v:0:21997:289:290:9004 +RTPI:10786:v:0:21997:300:301:9006 +TV7:10786:v:0:21997:166:128:9007 +ARTE:10786:v:0:21997:167:136:9009 +Colourbars:12611:v:0:22000:48:49:3982 +Alice:12611:v:0:22000:162:96:12200 +Video Italia:12611:v:0:22000:121:122:12220 +ANDALUCIA TV:11934:v:0:27500:166:104:29011 +TVC INT.:12441:v:0:27500:512:660:29701 +TV4:11992:h:0:27500:165:98:20365 +TV Niepokalanow:11876:h:0:27500:161:82:20601 +VIVA:12670:v:0:22000:309:310:12732 +VIVA ZWEI:12552:v:0:22000:171:172:12120 +MTV Central:12699:v:0:22000:3031:3032:28643 +ONYX:12692:h:1:27500:161:84:502 +VIVA polska:11603:h:1:27500:190:191:611 +DeeJay TV:11603:h:1:27500:160:161:602 +NBC:11053:h:1:27500:550:551:8008 +EWTN:10722:h:1:29900:1001:1201:4601 +MTA INTL:10722:h:1:29900:1004:1204:4604 +VOX:11053:h:1:27500:500:501:8002 +SAT.1 A:11053:h:1:27500:511:512:8003 +RTL2 AUSTRIA:11053:h:1:27500:520:521:8004 +ZDF:11053:h:1:27500:570:571:8011 +K-TV:11053:h:1:27500:580:581:8012 +RTL Television:11053:h:1:27500:160:80:8001 +ARTE:11059:v:1:6510:98:99:1 +HOT Italia:11095:h:1:27500:4194:4195:3714 +Olisat:11095:h:1:27500:33:34:3718 +VIVA-POLSKA:11128:h:1:4340:98:99:1 +DW-tv:11195:v:1:9099:101:102:5301 +Canal 24 Horas:11203:h:1:3999:4130:4131:5301 +TV5:11337:v:1:5631:512:640:1 +SAT.1 CH:11603:h:1:27500:101:102:601 +KurdSat:11603:h:1:27500:111:112:603 +ARD "Das Erste":11603:h:1:27500:172:173:606 +RTL 2 CH:11603:h:1:27500:175:176:609 +Super RTL A:11603:h:1:27500:180:181:610 +TV ROMANIA:11622:v:1:27500:227:247:10707 +MRTV:11622:v:1:27500:222:242:10702 +102.5 HIT Ch:11622:v:1:27500:224:244:10704 +TLC SAT:11622:v:1:27500:225:245:10705 +PRO-SAT:11622:v:1:27500:246:226:10706 +Channel SUN:11622:v:1:27500:229:249:10709 +Racing Channel:11622:v:1:27500:228:248:10708 +3 ABN:11622:v:1:27500:221:241:10701 +Bloom.Germany:11642:h:1:27500:1460:1420:4 +Bloomberg TV UK:11642:h:1:27500:1560:1520:4 +Sat 7:11642:h:1:27500:1660:1620:4 +EDTV 1:11746:h:1:27500:4130:4131:9501 +EDTV SPORT:11746:h:1:27500:4386:4387:9502 +EDTV BUSINESS:11746:h:1:27500:4642:4643:9503 +EDTV DRAMA:11746:h:1:27500:4898:4899:9504 +RAI1:11765:v:1:27499:160:80:3401 +RAI2:11765:v:1:27499:161:84:3402 +RAI3:11765:v:1:27499:162:88:3403 +RaiWayTEST2:11765:v:1:27499:516:654:3405 +RAIMOSAICO:11765:v:1:27499:518:8191:3407 +RAINews24:11803:v:1:27500:516:654:3301 +CAMERA DEPUTATI:11803:v:1:27500:517:655:3302 +TELEPACE:11803:v:1:27500:515:653:3304 +RAISPORTSAT:11803:v:1:27500:512:650:3305 +RAINettunoSAT2:11803:v:1:27500:513:651:3306 +RAIeducational:11803:v:1:27500:514:652:3307 +RAINettunoSAT1:11803:v:1:27500:519:657:3308 +SAT2000:11803:v:1:27500:518:656:3309 +I1:11918:v:1:27499:512:650:1 +C5:11918:v:1:27499:513:660:2 +R4:11918:v:1:27499:514:670:3 +Telesierra:12091:h:1:27500:4160:4161:8704 +C. Milagro:12091:h:1:27500:4368:4369:8711 +Italia Sat:12091:h:1:27500:4600:4601:8728 +TVE Internacional:12091:h:1:27500:4208:4209:8707 +Fiesta:12091:h:1:27500:4432:4433:8720 +Retelsat:12091:h:1:27500:4464:4465:8722 +ART EUROPE:12013:h:1:27495:164:96:450 +EGYPT SAT. CH. 2:12013:h:1:27495:166:104:470 +IQRA:12013:h:1:27495:168:112:474 +MAURITANIA TV:12110:v:1:27500:230:231:704 +ARMENIA TV:12110:v:1:27500:240:241:705 +SAILING CHANNEL:12110:v:1:27500:260:261:707 +AL JAZEERA:12110:v:1:27500:270:271:708 +Coming Soon TV:12110:v:1:27500:310:311:717 +SaluteBenessere:12110:v:1:27500:320:321:718 +AH-EDP1:12148:v:1:27499:96:97:7201 +AH-EDP2:12148:v:1:27499:112:113:7202 +Espresso:12148:v:1:27499:192:193:7203 +Alice:12148:v:1:27499:160:161:7220 +Nuvolari:12148:v:1:27499:176:177:7221 +Leonardo:12148:v:1:27499:128:129:7222 +AH-EDP3:12148:v:1:27499:36:37:7205 +OTE Promo:12187:v:1:27500:517:655:1001 +RTS SAT:12187:v:1:27500:519:657:1022 +ERT SAT:12187:v:1:27500:514:652:1102 +EXTRA:12187:v:1:27500:516:654:1106 +TRIAL:12187:v:1:27500:513:651:1108 +Minimax:11303:h:1:19540:300:301:3 +TVN1:12209:h:1:5631:4194:4195:1 +RR TEST:10978:v:1:8998:33:34:1 +TV 5 Thailand:10978:v:1:8998:1057:1058:2 +TEST-1:10978:v:1:8998:3105:3106:4 +FASHION:12244:h:1:27500:123:133:103 +AJARA TV:12244:h:1:27500:127:137:107 +SLO-TV1:12300:v:1:27495:200:201:3201 +POLONIA 1:12302:v:1:27500:205:206:3203 +SUPER 1:12302:v:1:27500:207:208:3207 +NAPOLI INT.:12302:v:1:27500:240:241:3210 +MAGIC:12302:v:1:27500:245:246:3211 +COUNTDOWN:12302:v:1:27500:235:236:3212 +TBNE:12302:v:1:27500:230:231:3213 +NAPOLI CHANNEL:12302:v:1:27500:227:228:3215 +KURDISTAN TV:12302:v:1:27500:225:226:3214 +ATLAS TV:12379:v:1:27500:3022:3032:3002 +TELE 24 SWITZERLAND:12379:v:1:27500:3023:3033:3003 +Abu Dhabi TV:12379:v:1:27500:3024:3034:3004 +RTV MONTENEGRO:12379:v:1:27500:3026:3036:3006 +JAAM-E-JAM 1:12436:h:1:27500:160:80:1 +JAAM-E-JAM 2:12436:h:1:27500:161:82:2 +SAHAR:12436:h:1:27500:162:84:3 +SAHAR 2:12436:h:1:27500:163:86:4 +IRINN:12436:h:1:27500:164:88:5 +Musicmax:11303:h:1:19540:500:501:6 +TEST:12474:h:1:27500:771:8191:10608 +EbS:12474:h:1:27500:101:201:10601 +MOU.2:12474:h:1:27500:42:43:10602 +PINK PLUS:12474:h:1:27500:308:256:10605 +LibertyTV.com:12474:h:1:27500:941:942:10603 +2M Maroc:12474:h:1:27500:601:602:10607 +ZEE TV:12474:h:1:27500:910:911:10604 +WorldNet Europe:12483:v:1:8299:4260:4220:1 +WorldNet:12483:v:1:8299:4560:4520:4 +SICILIA INTERNATIONA:12519:v:1:27499:501:502:8309 +SARDEGNA UNO:12519:v:1:27499:503:504:8310 +EuroMed:12519:v:1:27499:510:511:8312 +TGRT:12519:v:1:27499:505:506:8313 +VIDEOLINA:12519:v:1:27499:515:516:8318 +MEDIOLANUM:12538:h:1:27500:1131:1132:8987 +www.travel:12538:h:1:27500:1180:1183:8992 +MonteCarloSat:12538:h:1:27500:5126:5122:8877 +Bulgaria TV:12538:h:1:27500:4612:4613:8827 +TVN1:12571:h:1:5631:4194:4195:1 +JSTV 1:12595:v:1:27500:2000:2001:8213 +JSTV 2:12595:v:1:27500:2011:2013:8214 +MBC:12595:v:1:27500:160:80:8201 +ANN:12595:v:1:27500:161:84:8202 +BET:12595:v:1:27500:167:108:8208 +EuroNews:12595:v:1:27500:2221:2231:8211 +Sharjah        Arabs:12653:h:1:27500:1160:1120:1 +Qatar          Arabs:12653:h:1:27500:1260:1220:2 +Saudi 1        Arabs:12653:h:1:27500:1360:1320:3 +Kuwait         Arabs:12653:h:1:27500:1460:1420:4 +Libya          Arabs:12653:h:1:27500:1560:1520:5 +Sudan          Arabs:12653:h:1:27500:1660:1620:6 +Oman           Arabs:12653:h:1:27500:1760:1720:7 +Jordan         Arabs:12653:h:1:27500:1860:1820:8 +IRAQ TV:12653:h:1:27500:1960:1920:9 +Dubai Sport:12653:h:1:27500:1060:1020:10 +Digitaly:12672:v:1:27500:220:221:4203 +Telemarket:12672:v:1:27500:350:351:4211 +eVision:12672:v:1:27500:360:361:4214 +Thai TV5:12672:v:1:27500:200:201:4201 +Studio Europa:12672:v:1:27500:230:231:4204 +Video  Italia:12672:v:1:27500:340:341:4210 +GAME NETWORK:12672:v:1:27500:291:292:4213 +BBC-Choice:578000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:620:621 +BBC-Knowledge:578000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:630:631 +BBC-News24:578000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:640:641 +BBC-1:578000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:600:601 +BBC-Parliament:578000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:0:660 +BBC-2:578000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:610:611 +ITV-1:850000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:513:651 +ITV-2:850000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2818:2819 +ITV-Sport:850000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2836:2837 +FilmFour:850000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2822:2823 +C4:850000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2826:2827 +E4:850000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2831:2832 +C5:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6017:6018 +Shop:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6049:6050 +ITVSelect-Info:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6065:6066 +ITVSelect-1:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6081:6082 +ITVSelect-2:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6097:6098 +ITVSelect-3:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6113:6114 +ITVSelect-4:713833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:6129:6130 +Carlton-Cinema:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:288:289 +Sky-One:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:544:545 +Sky-Sports-1:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:800:801 +Sky-Premier:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:1056:1057 +CartoonNetwork:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:1312:1313 +UK-Horizons:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2336:2337 +ITV-Sport-Plus:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2592:2593 +ITVSportSelect:721833334:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:0:2596 +BreezeMen&Mtrs:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:800:801 +Granada-Plus:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:544:545 +MTV:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:1568:1569 +Sky-Movie-Max:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:1312:1313 +Sky-Sports-2:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2080:2081 +UK-Gold:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:288:289 +Wellbeing:690000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:1824:1825 +PLAY-uk:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:288:289 +UK-Style:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:544:545 +no-name:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:800:801 +Discovery:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:1312:1313 +Nick/Paramount:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2080:2081 +Sky-Sports-3:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2336:2337 +Brit-Eurosport:538000000:INVERSION_OFF:BANDWIDTH_8_MHZ:FEC_2_3:FEC_NONE:QAM_64:TRANSMISSION_MODE_2K:GUARD_INTERVAL_1_32:HIERARCHY_NONE:2592:2593 diff --git a/test/libdvben50221/Makefile b/test/libdvben50221/Makefile new file mode 100644 index 0000000..cd29679 --- /dev/null +++ b/test/libdvben50221/Makefile @@ -0,0 +1,14 @@ +# Makefile for linuxtv.org dvb-apps/test/libdvben50221 + +binaries = test-app       \ +           test-session   \ +           test-transport + +CPPFLAGS += -I../../lib +LDLIBS   += ../../lib/libdvbapi/libdvbapi.a ../../lib/libdvben50221/libdvben50221.a ../../lib/libucsi/libucsi.a -lpthread + +.PHONY: all + +all: $(binaries) + +include ../../Make.rules diff --git a/test/libdvben50221/test-app.c b/test/libdvben50221/test-app.c new file mode 100644 index 0000000..d99ec01 --- /dev/null +++ b/test/libdvben50221/test-app.c @@ -0,0 +1,854 @@ +/* +    en50221 encoder An implementation for libdvb +    an implementation for the en50221 transport layer + +    Copyright (C) 2004, 2005 Manu Abraham (manu@kromtek.com) +    Copyright (C) 2005 Julian Scheel (julian at jusst dot de) +    Copyright (C) 2006 Andrew de Quincey (adq_dvb@lidskialf.net) + +    This library 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 Lesser General Public License for more details. + +    You should have received a copy of the GNU Lesser General Public +    License along with this library; if not, write to the Free Software +    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA +*/ + +#include <stdio.h> +#include <ctype.h> +#include <unistd.h> +#include <libdvben50221/en50221_session.h> +#include <libdvben50221/en50221_app_utils.h> +#include <libdvben50221/en50221_app_ai.h> +#include <libdvben50221/en50221_app_auth.h> +#include <libdvben50221/en50221_app_ca.h> +#include <libdvben50221/en50221_app_datetime.h> +#include <libdvben50221/en50221_app_dvb.h> +#include <libdvben50221/en50221_app_epg.h> +#include <libdvben50221/en50221_app_lowspeed.h> +#include <libdvben50221/en50221_app_mmi.h> +#include <libdvben50221/en50221_app_rm.h> +#include <libdvben50221/en50221_app_smartcard.h> +#include <libdvben50221/en50221_app_teletext.h> +#include <libdvbapi/dvbca.h> +#include <pthread.h> +#include <libdvbcfg/dvbcfg_zapchannel.h> +#include <libdvbapi/dvbdemux.h> +#include <libucsi/section.h> +#include <libucsi/mpeg/section.h> + +#define DEFAULT_SLOT 0 + +#define MAX_SESSIONS 256 +#define MAX_TC 32 + +void *stackthread_func(void* arg); +void *pmtthread_func(void* arg); +int test_lookup_callback(void *arg, uint8_t slot_id, uint32_t requested_resource_id, +                         en50221_sl_resource_callback *callback_out, void **arg_out, uint32_t *connected_resource_id); +int test_session_callback(void *arg, int reason, uint8_t slot_id, uint16_t session_number, uint32_t resource_id); + +int test_datetime_enquiry_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint8_t response_interval); + +int test_rm_enq_callback(void *arg, uint8_t slot_id, uint16_t session_number); +int test_rm_reply_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint32_t resource_id_count, uint32_t *resource_ids); +int test_rm_changed_callback(void *arg, uint8_t slot_id, uint16_t session_number); + +int test_ai_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                     uint8_t application_type, uint16_t application_manufacturer, +                     uint16_t manufacturer_code, uint8_t menu_string_length, +                     uint8_t *menu_string); + +int test_ca_info_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint32_t ca_id_count, uint16_t *ca_ids); +int test_ca_pmt_reply_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                               struct en50221_app_pmt_reply *reply, uint32_t reply_size); + +int test_mmi_close_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint8_t cmd_id, uint8_t delay); + +int test_mmi_display_control_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t cmd_id, uint8_t mmi_mode); + +int test_mmi_keypad_control_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t cmd_id, uint8_t *key_codes, uint32_t key_codes_count); + +int test_mmi_subtitle_segment_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t *segment, uint32_t segment_size); + +int test_mmi_scene_end_mark_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t decoder_continue_flag, uint8_t scene_reveal_flag, +                                        uint8_t send_scene_done, uint8_t scene_tag); + +int test_mmi_scene_control_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                    uint8_t decoder_continue_flag, uint8_t scene_reveal_flag, +                                    uint8_t scene_tag); + +int test_mmi_subtitle_download_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t *segment, uint32_t segment_size); + +int test_mmi_flush_download_callback(void *arg, uint8_t slot_id, uint16_t session_number); + +int test_mmi_enq_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                            uint8_t blind_answer, uint8_t expected_answer_length, +                            uint8_t *text, uint32_t text_size); + +int test_mmi_menu_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                            struct en50221_app_mmi_text *title, +                            struct en50221_app_mmi_text *sub_title, +                            struct en50221_app_mmi_text *bottom, +                            uint32_t item_count, struct en50221_app_mmi_text *items, +                            uint32_t item_raw_length, uint8_t *items_raw); + +int test_app_mmi_list_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                struct en50221_app_mmi_text *title, +                                struct en50221_app_mmi_text *sub_title, +                                struct en50221_app_mmi_text *bottom, +                                uint32_t item_count, struct en50221_app_mmi_text *items, +                                uint32_t item_raw_length, uint8_t *items_raw); + +struct section_ext *read_section_ext(char *buf, int buflen, int adapter, int demux, int pid, int table_id); + + + + + + + +int adapterid; + +int shutdown_stackthread = 0; +int shutdown_pmtthread = 0; +int in_menu = 0; +int in_enq = 0; +int ca_connected = 0; +int pmt_pid = -1; +int ca_session_number = 0; + + +// instances of resources we actually implement here +struct en50221_app_rm *rm_resource; +struct en50221_app_datetime *datetime_resource; +struct en50221_app_ai *ai_resource; +struct en50221_app_ca *ca_resource; +struct en50221_app_mmi *mmi_resource; + +// lookup table used in resource manager implementation +struct resource { +    struct en50221_app_public_resource_id resid; +    uint32_t binary_resource_id; +    en50221_sl_resource_callback callback; +    void *arg; +}; +struct resource resources[20]; +int resources_count = 0; + +// this contains all known resource ids so we can see if the cam asks for something exotic +uint32_t resource_ids[] = { EN50221_APP_TELETEXT_RESOURCEID, +                            EN50221_APP_SMARTCARD_RESOURCEID(1), +                            EN50221_APP_RM_RESOURCEID, +                            EN50221_APP_MMI_RESOURCEID, +                            EN50221_APP_LOWSPEED_RESOURCEID(1,1), +                            EN50221_APP_EPG_RESOURCEID(1), +                            EN50221_APP_DVB_RESOURCEID, +                            EN50221_APP_CA_RESOURCEID, +                            EN50221_APP_DATETIME_RESOURCEID, +                            EN50221_APP_AUTH_RESOURCEID, +                            EN50221_APP_AI_RESOURCEID, }; +int resource_ids_count = sizeof(resource_ids)/4; + + +uint16_t ai_session_numbers[5]; + +uint16_t mmi_session_number; + +int main(int argc, char * argv[]) +{ +    pthread_t stackthread; +    pthread_t pmtthread; +    struct en50221_app_send_functions sendfuncs; + +    if ((argc < 2) || (argc > 3)) { +        fprintf(stderr, "Syntax: test-app <adapterid> [<pmtpid>]\n"); +        exit(1); +    } +    adapterid = atoi(argv[1]); +    if (argc == 3) { +        if (sscanf(argv[2], "%i", &pmt_pid) != 1) { +            fprintf(stderr, "Unable to parse PMT PID\n"); +            exit(1); +        } +    } + +    // create transport layer +    struct en50221_transport_layer *tl = en50221_tl_create(5, 32); +    if (tl == NULL) { +        fprintf(stderr, "Failed to create transport layer\n"); +        exit(1); +    } + +    // find CAMs +    int cafd; +    if (((cafd = dvbca_open(adapterid, 0)) < 0) || (dvbca_get_cam_state(cafd, DEFAULT_SLOT) == DVBCA_CAMSTATE_MISSING)) { +        fprintf(stderr, "Unable to open CAM on adapter %i\n", adapterid); +        exit(1); +    } + +    // reset it and wait +    dvbca_reset(cafd, DEFAULT_SLOT); +    printf("Found a CAM on adapter%i... waiting...\n", adapterid); +    while(dvbca_get_cam_state(cafd, DEFAULT_SLOT) != DVBCA_CAMSTATE_READY) { +        usleep(1000); +    } + +    // register it with the CA stack +    int slot_id = 0; +    if ((slot_id = en50221_tl_register_slot(tl, cafd, DEFAULT_SLOT, 1000, 100)) < 0) { +        fprintf(stderr, "Slot registration failed\n"); +        exit(1); +    } +    printf("slotid: %i\n", slot_id); + +    // create session layer +    struct en50221_session_layer *sl = en50221_sl_create(tl, 256); +    if (sl == NULL) { +        fprintf(stderr, "Failed to create session layer\n"); +        exit(1); +    } + +    // create the sendfuncs +    sendfuncs.arg        = sl; +    sendfuncs.send_data  = (en50221_send_data) en50221_sl_send_data; +    sendfuncs.send_datav = (en50221_send_datav) en50221_sl_send_datav; + +    // create the resource manager resource +    rm_resource = en50221_app_rm_create(&sendfuncs); +    en50221_app_decode_public_resource_id(&resources[resources_count].resid, EN50221_APP_RM_RESOURCEID); +    resources[resources_count].binary_resource_id = EN50221_APP_RM_RESOURCEID; +    resources[resources_count].callback = (en50221_sl_resource_callback) en50221_app_rm_message; +    resources[resources_count].arg = rm_resource; +    en50221_app_rm_register_enq_callback(rm_resource, test_rm_enq_callback, NULL); +    en50221_app_rm_register_reply_callback(rm_resource, test_rm_reply_callback, NULL); +    en50221_app_rm_register_changed_callback(rm_resource, test_rm_changed_callback, NULL); +    resources_count++; + +    // create the datetime resource +    datetime_resource = en50221_app_datetime_create(&sendfuncs); +    en50221_app_decode_public_resource_id(&resources[resources_count].resid, EN50221_APP_DATETIME_RESOURCEID); +    resources[resources_count].binary_resource_id = EN50221_APP_DATETIME_RESOURCEID; +    resources[resources_count].callback = (en50221_sl_resource_callback) en50221_app_datetime_message; +    resources[resources_count].arg = datetime_resource; +    en50221_app_datetime_register_enquiry_callback(datetime_resource, test_datetime_enquiry_callback, NULL); +    resources_count++; + +    // create the application information resource +    ai_resource = en50221_app_ai_create(&sendfuncs); +    en50221_app_decode_public_resource_id(&resources[resources_count].resid, EN50221_APP_AI_RESOURCEID); +    resources[resources_count].binary_resource_id = EN50221_APP_AI_RESOURCEID; +    resources[resources_count].callback = (en50221_sl_resource_callback) en50221_app_ai_message; +    resources[resources_count].arg = ai_resource; +    en50221_app_ai_register_callback(ai_resource, test_ai_callback, NULL); +    resources_count++; + +    // create the CA resource +    ca_resource = en50221_app_ca_create(&sendfuncs); +    en50221_app_decode_public_resource_id(&resources[resources_count].resid, EN50221_APP_CA_RESOURCEID); +    resources[resources_count].binary_resource_id = EN50221_APP_CA_RESOURCEID; +    resources[resources_count].callback = (en50221_sl_resource_callback) en50221_app_ca_message; +    resources[resources_count].arg = ca_resource; +    en50221_app_ca_register_info_callback(ca_resource, test_ca_info_callback, NULL); +    en50221_app_ca_register_pmt_reply_callback(ca_resource, test_ca_pmt_reply_callback, NULL); +    resources_count++; + +    // create the MMI resource +    mmi_resource = en50221_app_mmi_create(&sendfuncs); +    en50221_app_decode_public_resource_id(&resources[resources_count].resid, EN50221_APP_MMI_RESOURCEID); +    resources[resources_count].binary_resource_id = EN50221_APP_MMI_RESOURCEID; +    resources[resources_count].callback = (en50221_sl_resource_callback) en50221_app_mmi_message; +    resources[resources_count].arg = mmi_resource; +    en50221_app_mmi_register_close_callback(mmi_resource, test_mmi_close_callback, NULL); +    en50221_app_mmi_register_display_control_callback(mmi_resource, test_mmi_display_control_callback, NULL); +    en50221_app_mmi_register_keypad_control_callback(mmi_resource, test_mmi_keypad_control_callback, NULL); +    en50221_app_mmi_register_subtitle_segment_callback(mmi_resource, test_mmi_subtitle_segment_callback, NULL); +    en50221_app_mmi_register_scene_end_mark_callback(mmi_resource, test_mmi_scene_end_mark_callback, NULL); +    en50221_app_mmi_register_scene_control_callback(mmi_resource, test_mmi_scene_control_callback, NULL); +    en50221_app_mmi_register_subtitle_download_callback(mmi_resource, test_mmi_subtitle_download_callback, NULL); +    en50221_app_mmi_register_flush_download_callback(mmi_resource, test_mmi_flush_download_callback, NULL); +    en50221_app_mmi_register_enq_callback(mmi_resource, test_mmi_enq_callback, NULL); +    en50221_app_mmi_register_menu_callback(mmi_resource, test_mmi_menu_callback, NULL); +    en50221_app_mmi_register_list_callback(mmi_resource, test_app_mmi_list_callback, NULL); +    resources_count++; + +    // start another thread running the stack +    pthread_create(&stackthread, NULL, stackthread_func, tl); + +    // start another thread parsing PMT +    if (pmt_pid != -1) { +        pthread_create(&pmtthread, NULL, pmtthread_func, tl); +    } + +    // register callbacks +    en50221_sl_register_lookup_callback(sl, test_lookup_callback, sl); +    en50221_sl_register_session_callback(sl, test_session_callback, sl); + +    // create a new connection on each slot +    int tc = en50221_tl_new_tc(tl, slot_id); +    printf("tcid: %i\n", tc); + +    printf("Press a key to enter menu\n"); +    getchar(); +    en50221_app_ai_entermenu(ai_resource, ai_session_numbers[slot_id]); + +    // wait +    char tmp[256]; +    while(1) { +        fgets(tmp, sizeof(tmp), stdin); +        int choice = atoi(tmp); + +        if (in_menu) { +            en50221_app_mmi_menu_answ(mmi_resource, mmi_session_number, choice); +            in_menu = 0; +        } +        if (in_enq) { +            uint32_t i; +            uint32_t len = strlen(tmp); +            for(i=0; i< len; i++) { +                if (!isdigit(tmp[i])) { +                    len = i; +                    break; +                } +            } +	    en50221_app_mmi_answ(mmi_resource, mmi_session_number, MMI_ANSW_ID_ANSWER, (uint8_t*) tmp, len); +            in_enq = 0; +        } +    } +    printf("Press a key to exit\n"); +    getchar(); + +    // destroy slots +    en50221_tl_destroy_slot(tl, slot_id); +    shutdown_stackthread = 1; +    shutdown_pmtthread = 1; +    pthread_join(stackthread, NULL); +    if (pmt_pid != -1) { +        pthread_join(pmtthread, NULL); +    } + +    // destroy session layer +    en50221_sl_destroy(sl); + +    // destroy transport layer +    en50221_tl_destroy(tl); + +    return 0; +} + +int test_lookup_callback(void *arg, uint8_t slot_id, uint32_t requested_resource_id, +                         en50221_sl_resource_callback *callback_out, void **arg_out, uint32_t *connected_resource_id) +{ +    struct en50221_app_public_resource_id resid; +    (void)arg; + +    // decode the resource id +    if (en50221_app_decode_public_resource_id(&resid, requested_resource_id)) { +        printf("%02x:Public resource lookup callback %i %i %i\n", slot_id, +               resid.resource_class, resid.resource_type, resid.resource_version); +    } else { +        printf("%02x:Private resource lookup callback %08x\n", slot_id, requested_resource_id); +        return -1; +    } + +    // FIXME: need better comparison +    // FIXME: return resourceid we actually connected to + +    // try and find an instance of the resource +    int i; +    for(i=0; i<resources_count; i++) { +        if ((resid.resource_class == resources[i].resid.resource_class) && +            (resid.resource_type == resources[i].resid.resource_type)) { +            *callback_out = resources[i].callback; +            *arg_out = resources[i].arg; +            *connected_resource_id = resources[i].binary_resource_id; +            return 0; +        } +    } + +    return -1; +} + +int test_session_callback(void *arg, int reason, uint8_t slot_id, uint16_t session_number, uint32_t resource_id) +{ +    (void)arg; +    switch(reason) { +        case S_SCALLBACK_REASON_CAMCONNECTING: +            printf("%02x:CAM connecting to resource %08x, session_number %i\n", +                   slot_id, resource_id, session_number); +            break; +        case S_SCALLBACK_REASON_CAMCONNECTED: +            printf("%02x:CAM successfully connected to resource %08x, session_number %i\n", +                   slot_id, resource_id, session_number); + +            if (resource_id == EN50221_APP_RM_RESOURCEID) { +                en50221_app_rm_enq(rm_resource, session_number); +            } else if (resource_id == EN50221_APP_AI_RESOURCEID) { +                en50221_app_ai_enquiry(ai_resource, session_number); +            } else if (resource_id == EN50221_APP_CA_RESOURCEID) { +                en50221_app_ca_info_enq(ca_resource, session_number); +                ca_session_number = session_number; +            } + +            break; +        case S_SCALLBACK_REASON_CAMCONNECTFAIL: +            printf("%02x:CAM on failed to connect to resource %08x\n", slot_id, resource_id); +            break; +        case S_SCALLBACK_REASON_CONNECTED: +            printf("%02x:Host connection to resource %08x connected successfully, session_number %i\n", +                   slot_id, resource_id, session_number); +            break; +        case S_SCALLBACK_REASON_CONNECTFAIL: +            printf("%02x:Host connection to resource %08x failed, session_number %i\n", +                   slot_id, resource_id, session_number); +            break; +        case S_SCALLBACK_REASON_CLOSE: +            printf("%02x:Connection to resource %08x, session_number %i closed\n", +                   slot_id, resource_id, session_number); +            break; +        case S_SCALLBACK_REASON_TC_CONNECT: +            printf("%02x:Host originated transport connection %i connected\n", slot_id, session_number); +            break; +        case S_SCALLBACK_REASON_TC_CAMCONNECT: +            printf("%02x:CAM originated transport connection %i connected\n", slot_id, session_number); +            break; +    } +    return 0; +} + + + +int test_rm_enq_callback(void *arg, uint8_t slot_id, uint16_t session_number) +{ +    (void)arg; + +    printf("%02x:%s\n", slot_id, __func__); + +    if (en50221_app_rm_reply(rm_resource, session_number, resource_ids_count, resource_ids)) { +        printf("%02x:Failed to send reply to ENQ\n", slot_id); +    } + +    return 0; +} + +int test_rm_reply_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint32_t resource_id_count, uint32_t *_resource_ids) +{ +    (void)arg; +    printf("%02x:%s\n", slot_id, __func__); + +    uint32_t i; +    for(i=0; i< resource_id_count; i++) { +        printf("  CAM provided resource id: %08x\n", _resource_ids[i]); +    } + +    if (en50221_app_rm_changed(rm_resource, session_number)) { +        printf("%02x:Failed to send REPLY\n", slot_id); +    } + +    return 0; +} + +int test_rm_changed_callback(void *arg, uint8_t slot_id, uint16_t session_number) +{ +    (void)arg; +    printf("%02x:%s\n", slot_id, __func__); + +    if (en50221_app_rm_enq(rm_resource, session_number)) { +        printf("%02x:Failed to send ENQ\n", slot_id); +    } + +    return 0; +} + + + +int test_datetime_enquiry_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint8_t response_interval) +{ +    (void)arg; +    printf("%02x:%s\n", slot_id, __func__); +    printf("  response_interval:%i\n", response_interval); + +    if (en50221_app_datetime_send(datetime_resource, session_number, time(NULL), -1)) { +        printf("%02x:Failed to send datetime\n", slot_id); +    } + +    return 0; +} + + + +int test_ai_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                     uint8_t application_type, uint16_t application_manufacturer, +                     uint16_t manufacturer_code, uint8_t menu_string_length, +                     uint8_t *menu_string) +{ +    (void)arg; + +    printf("%02x:%s\n", slot_id, __func__); +    printf("  Application type: %02x\n", application_type); +    printf("  Application manufacturer: %04x\n", application_manufacturer); +    printf("  Manufacturer code: %04x\n", manufacturer_code); +    printf("  Menu string: %.*s\n", menu_string_length, menu_string); + +    ai_session_numbers[slot_id] = session_number; + +    return 0; +} + + + +int test_ca_info_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint32_t ca_id_count, uint16_t *ca_ids) +{ +    (void)arg; +    (void)session_number; + +    printf("%02x:%s\n", slot_id, __func__); +    uint32_t i; +    for(i=0; i< ca_id_count; i++) { +        printf("  Supported CA ID: %04x\n", ca_ids[i]); +    } + +    ca_connected = 1; +    return 0; +} + +int test_ca_pmt_reply_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                               struct en50221_app_pmt_reply *reply, uint32_t reply_size) +{ +    (void)arg; +    (void)session_number; +    (void)reply; +    (void)reply_size; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + + +int test_mmi_close_callback(void *arg, uint8_t slot_id, uint16_t session_number, uint8_t cmd_id, uint8_t delay) +{ +    (void)arg; +    (void)session_number; + +    printf("%02x:%s\n", slot_id, __func__); +    printf("  cmd_id: %02x\n", cmd_id); +    printf("  delay: %02x\n", delay); + +    return 0; +} + +int test_mmi_display_control_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t cmd_id, uint8_t mmi_mode) +{ +    (void)arg; +    (void)session_number; + +    printf("%02x:%s\n", slot_id, __func__); +    printf("  cmd_id: %02x\n", cmd_id); +    printf("  mode: %02x\n", mmi_mode); + +    if (cmd_id == MMI_DISPLAY_CONTROL_CMD_ID_SET_MMI_MODE) { +        struct en50221_app_mmi_display_reply_details details; + +        details.u.mode_ack.mmi_mode = mmi_mode; +        if (en50221_app_mmi_display_reply(mmi_resource, session_number, MMI_DISPLAY_REPLY_ID_MMI_MODE_ACK, &details)) { +            printf("%02x:Failed to send mode ack\n", slot_id); +        } +    } + +    return 0; +} + +int test_mmi_keypad_control_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t cmd_id, uint8_t *key_codes, uint32_t key_codes_count) +{ +    (void)arg; +    (void)session_number; +    (void)cmd_id; +    (void)key_codes; +    (void)key_codes_count; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + +int test_mmi_subtitle_segment_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t *segment, uint32_t segment_size) +{ +    (void)arg; +    (void)session_number; +    (void)segment; +    (void)segment_size; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + +int test_mmi_scene_end_mark_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t decoder_continue_flag, uint8_t scene_reveal_flag, +                                        uint8_t send_scene_done, uint8_t scene_tag) +{ +    (void)arg; +    (void)session_number; +    (void)decoder_continue_flag; +    (void)scene_reveal_flag; +    (void)send_scene_done; +    (void)scene_tag; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + +int test_mmi_scene_control_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                    uint8_t decoder_continue_flag, uint8_t scene_reveal_flag, +                                    uint8_t scene_tag) +{ +    (void)arg; +    (void)session_number; +    (void)decoder_continue_flag; +    (void)scene_reveal_flag; +    (void)scene_tag; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + +int test_mmi_subtitle_download_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                        uint8_t *segment, uint32_t segment_size) +{ +    (void)arg; +    (void)session_number; +    (void)segment; +    (void)segment_size; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + +int test_mmi_flush_download_callback(void *arg, uint8_t slot_id, uint16_t session_number) +{ +    (void)arg; +    (void)session_number; + +    printf("%02x:%s\n", slot_id, __func__); + +    return 0; +} + +int test_mmi_enq_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                            uint8_t blind_answer, uint8_t expected_answer_length, +                            uint8_t *text, uint32_t text_size) +{ +    (void)arg; +    (void)text; +    (void)text_size; + +    printf("%02x:%s\n", slot_id, __func__); +    printf("  blind: %i\n", blind_answer); +    printf("  expected_answer_length: %i\n", expected_answer_length); + +    mmi_session_number = session_number; +    in_enq = 1; + +    return 0; +} + +int test_mmi_menu_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                            struct en50221_app_mmi_text *title, +                            struct en50221_app_mmi_text *sub_title, +                            struct en50221_app_mmi_text *bottom, +                            uint32_t item_count, struct en50221_app_mmi_text *items, +                            uint32_t item_raw_length, uint8_t *items_raw) +{ +    (void)arg; +    (void)items_raw; + +    printf("%02x:%s\n", slot_id, __func__); + +    printf("  title: %.*s\n", title->text_length, title->text); +    printf("  sub_title: %.*s\n", sub_title->text_length, sub_title->text); +    printf("  bottom: %.*s\n", bottom->text_length, bottom->text); + +    uint32_t i; +    for(i=0; i< item_count; i++) { +        printf("  item %i: %.*s\n", i+1, items[i].text_length, items[i].text); +    } +    printf("  raw_length: %i\n", item_raw_length); + +    mmi_session_number = session_number; +    in_menu = 1; + +    return 0; +} + +int test_app_mmi_list_callback(void *arg, uint8_t slot_id, uint16_t session_number, +                                struct en50221_app_mmi_text *title, +                                struct en50221_app_mmi_text *sub_title, +                                struct en50221_app_mmi_text *bottom, +                                uint32_t item_count, struct en50221_app_mmi_text *items, +                                uint32_t item_raw_length, uint8_t *items_raw) +{ +    (void)arg; +    (void)items_raw; +    (void)arg; + +    printf("%02x:%s\n", slot_id, __func__); + +    printf("  title: %.*s\n", title->text_length, title->text); +    printf("  sub_title: %.*s\n", sub_title->text_length, sub_title->text); +    printf("  bottom: %.*s\n", bottom->text_length, bottom->text); + +    uint32_t i; +    for(i=0; i< item_count; i++) { +        printf("  item %i: %.*s\n", i+1, items[i].text_length, items[i].text); +    } +    printf("  raw_length: %i\n", item_raw_length); + +    mmi_session_number = session_number; +    in_menu = 1; + +    return 0; +} + + + + + + + +void *stackthread_func(void* arg) { +    struct en50221_transport_layer *tl = arg; +    int lasterror = 0; + +    while(!shutdown_stackthread) { +        int error; +        if ((error = en50221_tl_poll(tl)) != 0) { +            if (error != lasterror) { +                fprintf(stderr, "Error reported by stack slot:%i error:%i\n", +                        en50221_tl_get_error_slot(tl), +                        en50221_tl_get_error(tl)); +            } +            lasterror = error; +        } +    } + +    shutdown_stackthread = 0; +    return 0; +} + +void *pmtthread_func(void* arg) { +    (void)arg; +    char buf[4096]; +    uint8_t capmt[4096]; +    int pmtversion = -1; + +    while(!shutdown_pmtthread) { + +        if (!ca_connected) { +            sleep(1); +            continue; +        } + +        // read the PMT +        struct section_ext *section_ext = read_section_ext(buf, sizeof(buf), adapterid, 0, pmt_pid, stag_mpeg_program_map); +        if (section_ext == NULL) { +            fprintf(stderr, "Failed to read PMT\n"); +            exit(1); +        } +        struct mpeg_pmt_section *pmt = mpeg_pmt_section_codec(section_ext); +        if (pmt == NULL) { +            fprintf(stderr, "Bad PMT received\n"); +            exit(1); +        } +        if (pmt->head.version_number == pmtversion) { +            continue; +        } + +        // translate it into a CA PMT +        int listmgmt = CA_LIST_MANAGEMENT_ONLY; +        if (pmtversion != -1) { +            listmgmt = CA_LIST_MANAGEMENT_UPDATE; +        } +        int size; +        if ((size = en50221_ca_format_pmt(pmt, +                                         capmt, +                                         sizeof(capmt), +                                         listmgmt, +                                         0, +                                         CA_PMT_CMD_ID_OK_DESCRAMBLING)) < 0) { +            fprintf(stderr, "Failed to format CA PMT object\n"); +            exit(1); +        } + +        // set it +        if (en50221_app_ca_pmt(ca_resource, ca_session_number, capmt, size)) { +            fprintf(stderr, "Failed to send CA PMT object\n"); +            exit(1); +        } +        pmtversion = pmt->head.version_number; +    } +    shutdown_pmtthread = 0; +    return 0; +} + + +struct section_ext *read_section_ext(char *buf, int buflen, int adapter, int demux, int pid, int table_id) +{ +    int demux_fd = -1; +    uint8_t filter[18]; +    uint8_t mask[18]; +    int size; +    struct section *section; +    struct section_ext *result = NULL; + +    // open the demuxer +    if ((demux_fd = dvbdemux_open_demux(adapter, demux, 0)) < 0) { +        goto exit; +    } + +    // create a section filter +    memset(filter, 0, sizeof(filter)); +    memset(mask, 0, sizeof(mask)); +    filter[0] = table_id; +    mask[0] = 0xFF; +    if (dvbdemux_set_section_filter(demux_fd, pid, filter, mask, 1, 1)) { +        goto exit; +    } + +    // read the section +    if ((size = read(demux_fd, buf, buflen)) < 0) { +        goto exit; +    } + +    // parse it as a section +    section = section_codec((uint8_t*) buf, size); +    if (section == NULL) { +        goto exit; +    } + +    // parse it as a section_ext +    result = section_ext_decode(section, 0); + +exit: +    if (demux_fd != -1) +        close(demux_fd); +    return result; +} diff --git a/test/libdvben50221/test-session.c b/test/libdvben50221/test-session.c new file mode 100644 index 0000000..14dc13e --- /dev/null +++ b/test/libdvben50221/test-session.c @@ -0,0 +1,171 @@ +/* +    en50221 encoder An implementation for libdvb +    an implementation for the en50221 transport layer + +    Copyright (C) 2004, 2005 Manu Abraham (manu@kromtek.com) +    Copyright (C) 2005 Julian Scheel (julian at jusst dot de) +    Copyright (C) 2006 Andrew de Quincey (adq_dvb@lidskialf.net) + +    This library 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 Lesser General Public License for more details. + +    You should have received a copy of the GNU Lesser General Public +    License along with this library; if not, write to the Free Software +    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA +*/ + +#include <stdio.h> +#include <unistd.h> +#include <libdvben50221/en50221_session.h> +#include <libdvben50221/en50221_app_utils.h> +#include <libdvbapi/dvbca.h> +#include <pthread.h> + +void *stackthread_func(void* arg); +int test_lookup_callback(void *arg, uint8_t slot_id, uint32_t requested_resource_id, +                         en50221_sl_resource_callback *callback_out, void **arg_out, uint32_t *connected_resource_id); +int test_session_callback(void *arg, int reason, uint8_t slot_id, uint16_t session_number, uint32_t resource_id); + + +int shutdown_stackthread = 0; + +#define DEFAULT_SLOT 0 + +int main(int argc, char * argv[]) +{ +    (void)argc; +    (void)argv; + +    int i; +    pthread_t stackthread; + +    // create transport layer +    struct en50221_transport_layer *tl = en50221_tl_create(5, 32); +    if (tl == NULL) { +        fprintf(stderr, "Failed to create transport layer\n"); +        exit(1); +    } + +    // find CAMs +    int slot_count = 0; +    int cafd= -1; +    for(i=0; i<20; i++) { +        if ((cafd = dvbca_open(i, 0)) > 0) { +            if (dvbca_get_cam_state(cafd, DEFAULT_SLOT) == DVBCA_CAMSTATE_MISSING) { +                close(cafd); +                continue; +            } + +            // reset it and wait +            dvbca_reset(cafd, DEFAULT_SLOT); +            printf("Found a CAM on adapter%i... waiting...\n", i); +            while(dvbca_get_cam_state(cafd, DEFAULT_SLOT) != DVBCA_CAMSTATE_READY) { +                usleep(1000); +            } + +            // register it with the CA stack +            int slot_id = 0; +            if ((slot_id = en50221_tl_register_slot(tl, cafd, DEFAULT_SLOT, 1000, 100)) < 0) { +                fprintf(stderr, "Slot registration failed\n"); +                exit(1); +            } +            printf("slotid: %i\n", slot_id); +            slot_count++; +        } +    } + +    // create session layer +    struct en50221_session_layer *sl = en50221_sl_create(tl, 256); +    if (sl == NULL) { +        fprintf(stderr, "Failed to create session layer\n"); +        exit(1); +    } + +    // start another thread running the stack +    pthread_create(&stackthread, NULL, stackthread_func, tl); + +    // register callbacks +    en50221_sl_register_lookup_callback(sl, test_lookup_callback, sl); +    en50221_sl_register_session_callback(sl, test_session_callback, sl); + +    // create a new connection +    for(i=0; i<slot_count; i++) { +        int tc = en50221_tl_new_tc(tl, i); +        printf("tcid: %i\n", tc); +    } + +    // wait +    printf("Press a key to exit\n"); +    getchar(); + +    // destroy slots +    for(i=0; i<slot_count; i++) { +        en50221_tl_destroy_slot(tl, i); +    } +    shutdown_stackthread = 1; +    pthread_join(stackthread, NULL); + +    // destroy session layer +    en50221_sl_destroy(sl); + +    // destroy transport layer +    en50221_tl_destroy(tl); + +    return 0; +} + +int test_lookup_callback(void *arg, uint8_t slot_id, uint32_t requested_resource_id, +                         en50221_sl_resource_callback *callback_out, void **arg_out, uint32_t *connected_resource_id) +{ +    (void)arg; +    (void)callback_out; +    (void)arg_out; +    (void)connected_resource_id; + +    struct en50221_app_public_resource_id resid; + +    if (en50221_app_decode_public_resource_id(&resid, requested_resource_id)) { +        printf("Public resource lookup callback %i %i %i %i\n", slot_id, +               resid.resource_class, resid.resource_type, resid.resource_version); +    } else { +        printf("Private resource lookup callback %i %08x\n", slot_id, requested_resource_id); +    } + +    return -1; +} + +int test_session_callback(void *arg, int reason, uint8_t slot_id, uint16_t session_number, uint32_t resource_id) +{ +   (void)arg; + +    printf("Session callback %i %i %i %04x\n", slot_id, session_number, reason, resource_id); + +    return -1; +} + +void *stackthread_func(void* arg) { +    struct en50221_transport_layer *tl = arg; +    int lasterror = 0; + +    while(!shutdown_stackthread) { +        int error; +        if ((error = en50221_tl_poll(tl)) != 0) { +            if (error != lasterror) { +                fprintf(stderr, "Error reported by stack slot:%i error:%i\n", +                        en50221_tl_get_error_slot(tl), +                        en50221_tl_get_error(tl)); +            } +            lasterror = error; +        } +    } + +    shutdown_stackthread = 0; +    return 0; +} diff --git a/test/libdvben50221/test-transport.c b/test/libdvben50221/test-transport.c new file mode 100644 index 0000000..fdfd98a --- /dev/null +++ b/test/libdvben50221/test-transport.c @@ -0,0 +1,144 @@ +/* +    en50221 encoder An implementation for libdvb +    an implementation for the en50221 transport layer + +    Copyright (C) 2004, 2005 Manu Abraham (manu@kromtek.com) +    Copyright (C) 2005 Julian Scheel (julian at jusst dot de) +    Copyright (C) 2006 Andrew de Quincey (adq_dvb@lidskialf.net) + +    This library 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 Lesser General Public License for more details. + +    You should have received a copy of the GNU Lesser General Public +    License along with this library; if not, write to the Free Software +    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA +*/ + +#include <stdio.h> +#include <unistd.h> +#include <libdvben50221/en50221_transport.h> +#include <libdvbapi/dvbca.h> +#include <pthread.h> + +void *stackthread_func(void* arg); +void test_callback(void *arg, int reason, +                   uint8_t *data, uint32_t data_length, +                   uint8_t slot_id, uint8_t connection_id); + +int shutdown_stackthread = 0; + +#define DEFAULT_SLOT 0 + +int main(int argc, char * argv[]) +{ +    (void)argc; +    (void)argv; + +    int i; +    pthread_t stackthread; + +    // create transport layer +    struct en50221_transport_layer *tl = en50221_tl_create(5, 32); +    if (tl == NULL) { +        fprintf(stderr, "Failed to create transport layer\n"); +        exit(1); +    } + +    // find CAMs +    int slot_count = 0; +    int cafd= -1; +    for(i=0; i<20; i++) { +        if ((cafd = dvbca_open(i, 0)) > 0) { +            if (dvbca_get_cam_state(cafd, DEFAULT_SLOT) == DVBCA_CAMSTATE_MISSING) { +                close(cafd); +                continue; +            } + +            // reset it and wait +            dvbca_reset(cafd, DEFAULT_SLOT); +            printf("Found a CAM on adapter%i... waiting...\n", i); +            while(dvbca_get_cam_state(cafd, DEFAULT_SLOT) != DVBCA_CAMSTATE_READY) { +                usleep(1000); +            } + +            // register it with the CA stack +            int slot_id = 0; +            if ((slot_id = en50221_tl_register_slot(tl, cafd, DEFAULT_SLOT, 1000, 100)) < 0) { +                fprintf(stderr, "Slot registration failed\n"); +                exit(1); +            } +            printf("slotid: %i\n", slot_id); +            slot_count++; +        } +    } + +    // start another thread to running the stack +    pthread_create(&stackthread, NULL, stackthread_func, tl); + +    // register callback +    en50221_tl_register_callback(tl, test_callback, tl); + +    // create a new connection +    for(i=0; i<slot_count; i++) { +        int tc = en50221_tl_new_tc(tl, i); +        printf("tcid: %i\n", tc); +    } + +    // wait +    printf("Press a key to exit\n"); +    getchar(); + +    // destroy slots +    for(i=0; i<slot_count; i++) { +        en50221_tl_destroy_slot(tl, i); +    } +    shutdown_stackthread = 1; +    pthread_join(stackthread, NULL); + +    // destroy transport layer +    en50221_tl_destroy(tl); + +    return 0; +} + +void test_callback(void *arg, int reason, +                   uint8_t *data, uint32_t data_length, +                   uint8_t slot_id, uint8_t connection_id) +{ +    (void) arg; + +    printf("-----------------------------------\n"); +    printf("CALLBACK SLOTID:%i %i %i\n", slot_id, connection_id, reason); + +    uint32_t i; +    for(i=0; i< data_length; i++) { +        printf("%02x %02x\n", i, data[i]); +    } +} + +void *stackthread_func(void* arg) { +    struct en50221_transport_layer *tl = arg; +    int lasterror = 0; + +    while(!shutdown_stackthread) { +        int error; +        if ((error = en50221_tl_poll(tl)) != 0) { +            if (error != lasterror) { +                fprintf(stderr, "Error reported by stack slot:%i error:%i\n", +                        en50221_tl_get_error_slot(tl), +                        en50221_tl_get_error(tl)); +            } +            lasterror = error; +        } +    } + +    shutdown_stackthread = 0; +    return 0; +} diff --git a/test/libdvbsec/Makefile b/test/libdvbsec/Makefile new file mode 100644 index 0000000..27c9c21 --- /dev/null +++ b/test/libdvbsec/Makefile @@ -0,0 +1,12 @@ +# Makefile for linuxtv.org dvb-apps/test/libdvbsec + +binaries = dvbsec_test + +CPPFLAGS += -I../../lib +LDLIBS   += ../../lib/libdvbsec/libdvbsec.a + +.PHONY: all + +all: $(binaries) + +include ../../Make.rules diff --git a/test/libdvbsec/dvbsec_test.c b/test/libdvbsec/dvbsec_test.c new file mode 100644 index 0000000..b49d703 --- /dev/null +++ b/test/libdvbsec/dvbsec_test.c @@ -0,0 +1,85 @@ +/** + * dvbsec testing. + * + * Copyright (c) 2005 by Andrew de Quincey <adq_dvb@lidskialf.net> + * + * This library 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 Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA + */ + +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <libdvbsec/dvbsec_cfg.h> + +void syntax(void); + +struct dvbsec_config *secconfigs = NULL; +int seccount = 0; + +int secload_callback(void *private, struct dvbsec_config *sec); + +int main(int argc, char *argv[]) +{ +        if (argc != 4) { +                syntax(); +        } + +	if (!strcmp(argv[1], "-sec")) { + +		FILE *f = fopen(argv[2], "r"); +		if (!f) { +			fprintf(stderr, "Unable to load %s\n", argv[2]); +			exit(1); +		} +		dvbsec_cfg_load(f, NULL, secload_callback); +		fclose(f); + +		f = fopen(argv[3], "w"); +		if (!f) { +			fprintf(stderr, "Unable to write %s\n", argv[3]); +			exit(1); +		} +		dvbsec_cfg_save(f, secconfigs, seccount); +		fclose(f); + +	} else { +                syntax(); +        } + +        exit(0); +} + +int secload_callback(void *private, struct dvbsec_config *sec) +{ +	(void) private; + +	struct dvbsec_config *tmp = realloc(secconfigs, (seccount+1) * sizeof(struct dvbsec_config)); +	if (tmp == NULL) { +		fprintf(stderr, "Out of memory\n"); +		exit(1); +	} +	secconfigs = tmp; + +	memcpy(&secconfigs[seccount++], sec, sizeof(struct dvbsec_config)); + +	return 0; +} + +void syntax() +{ +        fprintf(stderr, +                "Syntax: dvbcfg_test <-zapchannel|-sec> <input filename> <output filename>\n"); +        exit(1); +} diff --git a/test/libdvbsec/test_sec.txt b/test/libdvbsec/test_sec.txt new file mode 100644 index 0000000..1189026 --- /dev/null +++ b/test/libdvbsec/test_sec.txt @@ -0,0 +1,28 @@ +[sec] +name=test1 +switch-frequency=100000 +lof-lo-v=20 +lof-lo-h=30 +lof-lo-l=40 +lof-lo-r=50 +lof-hi-v=60 +lof-hi-h=70 +lof-hi-l=80 +lof-hi-r=90 +config-type=none + +[sec] +name=test2 +switch-frequency=200 +lof-lo-v=40 +lof-hi-h=50 +config-type=simple + +[sec] +name=test3 +switch-frequency=100000 +lof-lo-l=20 +lof-hi-r=30 +config-type=advanced +cmd-lo-v=MOOVH +cmd-lo-h=MOOLH diff --git a/test/libesg/Makefile b/test/libesg/Makefile new file mode 100644 index 0000000..11d65bd --- /dev/null +++ b/test/libesg/Makefile @@ -0,0 +1,12 @@ +# Makefile for linuxtv.org dvb-apps/test/libucsi + +binaries = testesg + +CPPFLAGS += -I../../lib +LDLIBS   += ../../lib/libesg/libesg.a + +.PHONY: all + +all: $(binaries) + +include ../../Make.rules diff --git a/test/libesg/samples/ESGProviderDiscoveryDescriptor.xml b/test/libesg/samples/ESGProviderDiscoveryDescriptor.xml new file mode 100644 index 0000000..3989afb --- /dev/null +++ b/test/libesg/samples/ESGProviderDiscoveryDescriptor.xml @@ -0,0 +1,13 @@ +<?xml version="1.0"?> +<ESGProviderDiscovery xmlns="urn:dvb:ipdc:esgbs:2005" xmlns:mpeg7="urn:mpeg:mpeg7:schema:2001"> +  <ServiceProvider> +    <ProviderURI>http://www.sidsa.com</ProviderURI> +    <ProviderName>SIDSA (Multiple Stream)</ProviderName> +    <ProviderID>1</ProviderID> +  </ServiceProvider> +  <ServiceProvider> +    <ProviderURI>http://www.sidsa.com</ProviderURI> +    <ProviderName>SIDSA (Single Stream)</ProviderName> +    <ProviderID>2</ProviderID> +  </ServiceProvider> +</ESGProviderDiscovery> diff --git a/test/libesg/testesg.c b/test/libesg/testesg.c new file mode 100644 index 0000000..f2183ac --- /dev/null +++ b/test/libesg/testesg.c @@ -0,0 +1,563 @@ +/* + * ESG parser + * + * Copyright (C) 2006 Stephane Este-Gracias (sestegra@free.fr) + * + * This library 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 library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA + */ + +#include <stdio.h> +#include <unistd.h> +#include <stdlib.h> +#include <string.h> +#include <fcntl.h> +#include <sys/stat.h> + +#include <libesg/bootstrap/access_descriptor.h> +#include <libesg/encapsulation/container.h> +#include <libesg/encapsulation/fragment_management_information.h> +#include <libesg/encapsulation/data_repository.h> +#include <libesg/encapsulation/string_repository.h> +#include <libesg/representation/encapsulated_textual_esg_xml_fragment.h> +#include <libesg/representation/init_message.h> +#include <libesg/representation/textual_decoder_init.h> +#include <libesg/representation/bim_decoder_init.h> +#include <libesg/transport/session_partition_declaration.h> + +#define MAX_FILENAME 256 + +void usage(void) { +  static const char *_usage = +    "Usage: testesg [-a <ESGAccessDescriptor>]\n" +    "               [-c <ESGContainer with Textual ESG XML Fragment>]\n" +    "               [-X XXXX]\n"; + +  fprintf(stderr, "%s", _usage); +  exit(1); +} + +void read_from_file(const char *filename, char **buffer, int *size) { +	int fd; +	struct stat fs; + +	if ((fd = open(filename, O_RDONLY)) <= 0) { +		fprintf(stderr, "File not found\n"); +		exit(1); +	} + +	if (fstat(fd, &fs) < 0) { +		fprintf(stderr, "File not readable\n"); +		exit(1); +	} +	*size = fs.st_size; + +	*buffer = (char *) malloc(*size); +	if (read(fd, *buffer, *size) != *size) { +		fprintf(stderr, "File read error\n"); +		exit(1); +	} + +	close(fd); + +	return; +} + +int main(int argc, char *argv[]) { +	char access_descriptor_filename[MAX_FILENAME] = ""; +	char container_filename[MAX_FILENAME] = ""; +	int c; +	char *buffer = NULL; +	int size; + +	// Read command line options +	while ((c = getopt(argc, argv, "a:c:")) != -1) { +		switch (c) { +			case 'a': +				strncpy(access_descriptor_filename, optarg, MAX_FILENAME); +				break; +			case 'c': +				strncpy(container_filename, optarg, MAX_FILENAME); +				break; +			default: +				usage(); +		} +	} + +	// ESGAccessDescriptor +	if (strncmp(access_descriptor_filename, "", MAX_FILENAME) != 0) { +		fprintf(stdout, "**************************************************\n"); +		fprintf(stdout, "Reading ESG Access Descriptor = %s\n", access_descriptor_filename); +		fprintf(stdout, "**************************************************\n\n"); + +		read_from_file(access_descriptor_filename, &buffer, &size); + +		struct esg_access_descriptor *access_descriptor = esg_access_descriptor_decode((uint8_t *) buffer, size); +		free(buffer); +		if (access_descriptor == NULL) { +			fprintf(stderr, "ESG Access Descriptor decode error\n"); +			exit(1); +		} +		fprintf(stdout, "n_o_ESGEntries %d\n\n", access_descriptor->n_o_entries); + +		struct esg_entry *entry; +		esg_access_descriptor_entry_list_for_each(access_descriptor, entry) { +			fprintf(stdout, "    ESGEntryVersion %d\n", entry->version); +			fprintf(stdout, "    MultipleStreamTransport %d\n", entry->multiple_stream_transport); +			fprintf(stdout, "    IPVersion6 %d\n", entry->ip_version_6); +			fprintf(stdout, "    ProviderID %d\n", entry->provider_id); +			if (entry->ip_version_6 == 0) { +				fprintf(stdout, "    SourceIPAddress %d.%d.%d.%d\n", +					entry->source_ip.ipv4[0], +					entry->source_ip.ipv4[1], +					entry->source_ip.ipv4[2], +					entry->source_ip.ipv4[3]); +				fprintf(stdout, "    DestinationIPAddress %d.%d.%d.%d\n", +					entry->destination_ip.ipv4[0], +					entry->destination_ip.ipv4[1], +					entry->destination_ip.ipv4[2], +					entry->destination_ip.ipv4[3]); +			} else if (entry->ip_version_6 == 1) { +				fprintf(stdout, "    SourceIPAddress %02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x\n", +					entry->source_ip.ipv6[0], +					entry->source_ip.ipv6[1], +					entry->source_ip.ipv6[2], +					entry->source_ip.ipv6[3], +					entry->source_ip.ipv6[4], +					entry->source_ip.ipv6[5], +					entry->source_ip.ipv6[6], +					entry->source_ip.ipv6[7], +					entry->source_ip.ipv6[8], +					entry->source_ip.ipv6[9], +					entry->source_ip.ipv6[10], +					entry->source_ip.ipv6[11], +					entry->source_ip.ipv6[12], +					entry->source_ip.ipv6[13], +					entry->source_ip.ipv6[14], +					entry->source_ip.ipv6[15]); +				fprintf(stdout, "    DestinationIPAddress %02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x\n", +					entry->destination_ip.ipv6[0], +					entry->destination_ip.ipv6[1], +					entry->destination_ip.ipv6[2], +					entry->destination_ip.ipv6[3], +					entry->destination_ip.ipv6[4], +					entry->destination_ip.ipv6[5], +					entry->destination_ip.ipv6[6], +					entry->destination_ip.ipv6[7], +					entry->destination_ip.ipv6[8], +					entry->destination_ip.ipv6[9], +					entry->destination_ip.ipv6[10], +					entry->destination_ip.ipv6[11], +					entry->destination_ip.ipv6[12], +					entry->destination_ip.ipv6[13], +					entry->destination_ip.ipv6[14], +					entry->destination_ip.ipv6[15]); +				} +			fprintf(stdout, "Port %d\n", entry->port); +			fprintf(stdout, "TSI %d\n", entry->tsi); +			fprintf(stdout, "\n"); +		} +	} + +	// ESGContainer +	if (strncmp(container_filename, "", MAX_FILENAME) != 0) { +		fprintf(stdout, "**************************************************\n"); +		fprintf(stdout, "Reading ESG Container = %s\n", container_filename); +		fprintf(stdout, "**************************************************\n\n"); + +		read_from_file(container_filename, &buffer, &size); + +		struct esg_container *container = esg_container_decode((uint8_t *) buffer, size); +		free(buffer); +		if (container == NULL) { +			fprintf(stderr, "ESG Container decode error\n"); +			exit(1); +		} +		if (container->header == NULL) { +			fprintf(stderr, "ESG Container no header found\n"); +			exit(1); +		} + +		struct esg_encapsulation_structure *fragment_management_information = NULL; +		struct esg_data_repository *data_repository = NULL; +		struct esg_string_repository *string_repository = NULL; +		struct esg_container_structure *structure = NULL; +		struct esg_init_message *init_message = NULL; +		struct esg_textual_encoding_parameters *textual_encoding_parameters = NULL; +		struct esg_textual_decoder_init *textual_decoder_init = NULL; +		struct esg_namespace_prefix *namespace_prefix = NULL; +		struct esg_xml_fragment_type *xml_fragment_type = NULL; +		struct esg_bim_encoding_parameters *bim_encoding_parameters = NULL; +		struct esg_bim_decoder_init *bim_decoder_init = NULL; +		struct esg_session_partition_declaration *partition = NULL; +		struct esg_session_field *field = NULL; +		struct esg_session_ip_stream *ip_stream = NULL; +		struct esg_session_ip_stream_field *ip_stream_field = NULL; +		esg_container_header_structure_list_for_each(container->header, structure) { +			fprintf(stdout, "    structure_type %d [0x%02x]\n", structure->type, structure->type); +			fprintf(stdout, "    structure_id %d [0x%02x]\n", structure->id, structure->id); +			fprintf(stdout, "    structure_ptr %d\n", structure->ptr); +			fprintf(stdout, "    structure_length %d\n\n", structure->length); +				switch (structure->type) { +				case 0x01: { +					switch (structure->id) { +						case 0x00: { +							fprintf(stdout, "        ESG Fragment Management Information\n"); + +							fragment_management_information = (struct esg_encapsulation_structure *) structure->data; +							if (fragment_management_information == NULL) { +								fprintf(stderr, "ESG Fragment Management Information decode error\n"); +								exit(1); +							} + +							fprintf(stdout, "            fragment_reference_format %d [0x%02x]\n\n", fragment_management_information->header->fragment_reference_format, fragment_management_information->header->fragment_reference_format); + +							struct esg_encapsulation_entry *entry; +							esg_encapsulation_structure_entry_list_for_each(fragment_management_information, entry) { +								fprintf(stdout, "                fragment_type %d [0x%02x]\n", entry->fragment_reference->fragment_type, entry->fragment_reference->fragment_type); +								fprintf(stdout, "                data_repository_offset %d\n", entry->fragment_reference->data_repository_offset); +								fprintf(stdout, "                fragment_version %d\n", entry->fragment_version); +								fprintf(stdout, "                fragment_id %d\n\n", entry->fragment_id); +							} + +							break; +						} +						default: { +							fprintf(stdout, "        Unknown structure_id\n"); +						} +					} +					break; +				} +				case 0x02: { +					switch (structure->id) { +						case 0x00: { +							fprintf(stdout, "        ESG String Repository / "); + +							string_repository = (struct esg_string_repository *) structure->data; +							if (string_repository == NULL) { +								fprintf(stderr, "ESG String Repository decode error\n"); +								exit(1); +							} + +							fprintf(stdout, "encoding_type %d / length %d\n\n", string_repository->encoding_type, string_repository->length); + +							break; +						} +						default: { +							fprintf(stdout, "        Unknown structure_id\n"); +						} +					} +					break; +				} +				case 0x03: { +					//TODO +					break; +				} +				case 0x04: { +					//TODO +					break; +				} +				case 0x05: { +					//TODO +					break; +				} +				case 0xE0: { +					switch (structure->id) { +						case 0x00: { +							fprintf(stdout, "        ESG Data Repository / "); + +							data_repository = (struct esg_data_repository *) structure->data; +							if (data_repository == NULL) { +								fprintf(stderr, "ESG Data Repository decode error\n"); +								exit(1); +							} + +							fprintf(stdout, "length %d\n\n", data_repository->length); + +							break; +						} +						default: { +							fprintf(stdout, "        Unknown structure_id\n"); +						} +					} +					break; +				} +				case 0xE1: { +					switch (structure->id) { +						case 0xFF: { +							fprintf(stdout, "        ESG Session Partition Declaration\n"); + +							partition = (struct esg_session_partition_declaration *) structure->data; +							fprintf(stdout, "            num_fields %d\n", partition->num_fields); +							fprintf(stdout, "            overlapping %d\n\n", partition->overlapping); +							esg_session_partition_declaration_field_list_for_each(partition, field) { +								fprintf(stdout, "                identifier %d\n", field->identifier); +								fprintf(stdout, "                encoding %d\n",field->encoding); +								fprintf(stdout, "                length %d\n\n",field->length); +							} +							fprintf(stdout, "            n_o_IPStreams %d\n", partition->n_o_ip_streams); +							fprintf(stdout, "            IPVersion6 %d\n\n", partition->ip_version_6); +							esg_session_partition_declaration_ip_stream_list_for_each(partition, ip_stream) { +								fprintf(stdout, "                IPStreamID %d\n", ip_stream->id); +								if (partition->ip_version_6 == 0) { +									fprintf(stdout, "                SourceIPAddress %d.%d.%d.%d\n", +										ip_stream->source_ip.ipv4[0], +										ip_stream->source_ip.ipv4[1], +										ip_stream->source_ip.ipv4[2], +										ip_stream->source_ip.ipv4[3]); +									fprintf(stdout, "                DestinationIPAddress %d.%d.%d.%d\n", +										ip_stream->destination_ip.ipv4[0], +										ip_stream->destination_ip.ipv4[1], +										ip_stream->destination_ip.ipv4[2], +										ip_stream->destination_ip.ipv4[3]); +								} else if (partition->ip_version_6 == 1) { +									fprintf(stdout, "                SourceIPAddress %02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x\n", +										ip_stream->source_ip.ipv6[0], +										ip_stream->source_ip.ipv6[1], +										ip_stream->source_ip.ipv6[2], +										ip_stream->source_ip.ipv6[3], +										ip_stream->source_ip.ipv6[4], +										ip_stream->source_ip.ipv6[5], +										ip_stream->source_ip.ipv6[6], +										ip_stream->source_ip.ipv6[7], +										ip_stream->source_ip.ipv6[8], +										ip_stream->source_ip.ipv6[9], +										ip_stream->source_ip.ipv6[10], +										ip_stream->source_ip.ipv6[11], +										ip_stream->source_ip.ipv6[12], +										ip_stream->source_ip.ipv6[13], +										ip_stream->source_ip.ipv6[14], +										ip_stream->source_ip.ipv6[15]); +									fprintf(stdout, "                DestinationIPAddress %02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x:%02x%02x\n", +										ip_stream->destination_ip.ipv6[0], +										ip_stream->destination_ip.ipv6[1], +										ip_stream->destination_ip.ipv6[2], +										ip_stream->destination_ip.ipv6[3], +										ip_stream->destination_ip.ipv6[4], +										ip_stream->destination_ip.ipv6[5], +										ip_stream->destination_ip.ipv6[6], +										ip_stream->destination_ip.ipv6[7], +										ip_stream->destination_ip.ipv6[8], +										ip_stream->destination_ip.ipv6[9], +										ip_stream->destination_ip.ipv6[10], +										ip_stream->destination_ip.ipv6[11], +										ip_stream->destination_ip.ipv6[12], +										ip_stream->destination_ip.ipv6[13], +										ip_stream->destination_ip.ipv6[14], +										ip_stream->destination_ip.ipv6[15]); +								} +								fprintf(stdout, "                Port %d\n", ip_stream->port); +								fprintf(stdout, "                SessionID %d\n", ip_stream->session_id); + +								field = partition->field_list; +								esg_session_ip_stream_field_list_for_each(ip_stream, ip_stream_field) { +									switch (field->encoding) { +										case 0x0000: { +											if (ip_stream_field->start_field_value != NULL) { +												fprintf(stdout, "                start_field_value %s\n", ip_stream_field->start_field_value->string); +											} +											fprintf(stdout, "                end_field_value %s\n", ip_stream_field->end_field_value->string); +											break; +										} +										case 0x0101: { +											if (ip_stream_field->start_field_value != NULL) { +												fprintf(stdout, "                start_field_value %d\n", ip_stream_field->start_field_value->unsigned_short); +											} +											fprintf(stdout, "                end_field_value %d\n", ip_stream_field->end_field_value->unsigned_short); +											break; +										} +									} + +									field = field->_next; +								} +								fprintf(stdout, "\n"); +							} +							break; +						} +						default: { +							fprintf(stdout, "        Unknown structure_id\n"); +						} +					} +					break; +				} +				case 0xE2: { +					switch (structure->id) { +						case 0x00: { +							fprintf(stdout, "        ESG Init Message\n"); + +							init_message = (struct esg_init_message *) structure->data; +							if (init_message == NULL) { +								fprintf(stderr, "ESG Init Message decode error\n"); +								exit(1); +							} + +							fprintf(stdout, "            EncodingVersion %d [0x%02x]\n", init_message->encoding_version, init_message->encoding_version); +							fprintf(stdout, "            IndexingFlag %d\n", init_message->indexing_flag); +							fprintf(stdout, "            DecoderInitptr %d\n", init_message->decoder_init_ptr); +							if (init_message->indexing_flag) { +								fprintf(stdout, "            IndexingVersion %d\n", init_message->indexing_version); +							} + +							switch (init_message->encoding_version) { +								case 0xF1: { +									bim_encoding_parameters = (struct esg_bim_encoding_parameters *) init_message->encoding_parameters; +									if (bim_encoding_parameters == NULL) { +										fprintf(stderr, "ESG Init Message decode error / bim_encoding_parameters\n"); +										exit(1); +									} +									fprintf(stdout, "            BufferSizeFlag %d\n", bim_encoding_parameters->buffer_size_flag); +									fprintf(stdout, "            PositionCodeFlag %d\n", bim_encoding_parameters->position_code_flag); +									fprintf(stdout, "            CharacterEncoding %d\n", bim_encoding_parameters->character_encoding); +									if (bim_encoding_parameters->buffer_size_flag) { +										fprintf(stdout, "            BufferSize %d\n", bim_encoding_parameters->buffer_size); +									} + +									// TODO BimDecoderInit +									break; +								} +								case 0xF2: +								case 0xF3: { +									textual_encoding_parameters = (struct esg_textual_encoding_parameters *) init_message->encoding_parameters; +									if (textual_encoding_parameters == NULL) { +										fprintf(stderr, "ESG Init Message decode error / textual_encoding_parameters\n"); +										exit(1); +									} +									fprintf(stdout, "            CharacterEncoding %d\n\n", textual_encoding_parameters->character_encoding); + +									// TextualDecoderInit +									textual_decoder_init = (struct esg_textual_decoder_init *) init_message->decoder_init; +									if (textual_decoder_init == NULL) { +										fprintf(stderr, "ESG Init Message decode error / textual_decoder_init\n"); +										exit(1); +									} +									fprintf(stdout, "            Textual DecoderInit\n"); +									fprintf(stdout, "                num_namespaces_prefixes %d\n\n", textual_decoder_init->num_namespace_prefixes); +									esg_textual_decoder_namespace_prefix_list_for_each(textual_decoder_init, namespace_prefix) { +										fprintf(stdout, "                    prefix_string_ptr %d\n", namespace_prefix->prefix_string_ptr); +										fprintf(stdout, "                    namespace_URI_ptr %d\n\n", namespace_prefix->namespace_uri_ptr); +									} +									fprintf(stdout, "                num_fragment_types %d\n\n", textual_decoder_init->num_fragment_types); +									esg_textual_decoder_xml_fragment_type_list_for_each(textual_decoder_init, xml_fragment_type) { +										fprintf(stdout, "                    xpath_ptr %d\n", xml_fragment_type->xpath_ptr); +										fprintf(stdout, "                    ESG_XML_fragment_type %d\n\n", xml_fragment_type->xml_fragment_type); +									} +									break; +								} +								default: { +									fprintf(stdout, "            Unknown EncodingVersion\n"); +								} +							} + +							break; +						} +						default: { +							fprintf(stdout, "        Unknown structure_id\n"); +						} +					} +					break; +				} +				default: { +					fprintf(stdout, "        Unknown structure_type\n"); +				} +			} +		} +		fprintf(stdout, "\n"); + +		fprintf(stdout, "structure_body_ptr %d\n", container->structure_body_ptr); +		fprintf(stdout, "structure_body_length %d\n\n", container->structure_body_length); + +		// ESG XML Fragment +		if (fragment_management_information) { +			fprintf(stdout, "**************************************************\n"); +			fprintf(stdout, "ESG XML Fragment\n"); +			fprintf(stdout, "**************************************************\n\n"); + +			struct esg_encapsulation_entry *entry; +			esg_encapsulation_structure_entry_list_for_each(fragment_management_information, entry) { +				switch (entry->fragment_reference->fragment_type) { +					case 0x00: { +						if (data_repository) { +							struct esg_encapsulated_textual_esg_xml_fragment *esg_xml_fragment = esg_encapsulated_textual_esg_xml_fragment_decode(data_repository->data + entry->fragment_reference->data_repository_offset, data_repository->length); + +							fprintf(stdout, "ESG_XML_fragment_type %d\n", esg_xml_fragment->esg_xml_fragment_type); +							fprintf(stdout, "data_length %d\n", esg_xml_fragment->data_length); +							fprintf(stdout, "fragment_version %d\n", entry->fragment_version); +							fprintf(stdout, "fragment_id %d\n\n", entry->fragment_id); +							char *string = (char *) malloc(esg_xml_fragment->data_length + 1); +							memcpy(string, esg_xml_fragment->data, esg_xml_fragment->data_length); +							string[esg_xml_fragment->data_length] = 0; +							fprintf(stdout, "%s\n", string); + +						} else { +							fprintf(stderr, "ESG Data Repository not found"); +						} +						break; +					} +					case 0x01: { +						// TODO +						break; +					} +					case 0x02: { +						// TODO +						break; +					} +					default: { +					} +				} +			} +		} + +		// String +		if (init_message) { +			fprintf(stdout, "**************************************************\n"); +			fprintf(stdout, "String\n"); +			fprintf(stdout, "**************************************************\n\n"); + +			switch (init_message->encoding_version) { +				case 0xF1: { +					// TODO Bim +					break; +				} +				case 0xF2: { +					// TODO GZIP +					break; +				} +				case 0xF3: { +					// RAW +					if (string_repository) { +						textual_decoder_init = (struct esg_textual_decoder_init *) init_message->decoder_init; +						esg_textual_decoder_namespace_prefix_list_for_each(textual_decoder_init, namespace_prefix) { +							fprintf(stdout, "prefix_string_ptr %d\n", namespace_prefix->prefix_string_ptr); +							fprintf(stdout, "%s\n", string_repository->data + namespace_prefix->prefix_string_ptr); +							fprintf(stdout, "namespace_URI_ptr %d\n", namespace_prefix->namespace_uri_ptr); +							fprintf(stdout, "%s\n\n", string_repository->data + namespace_prefix->namespace_uri_ptr - 1); // TODO -1 +						} + +						esg_textual_decoder_xml_fragment_type_list_for_each(textual_decoder_init, xml_fragment_type) { +							fprintf(stdout, "xpath_ptr %d\n", xml_fragment_type->xpath_ptr); +							fprintf(stdout, "ESG_XML_fragment_type %d\n", xml_fragment_type->xml_fragment_type); +							fprintf(stdout, "%s\n\n", string_repository->data + xml_fragment_type->xpath_ptr - 1); // TODO -1 +						} +					} +					break; +				} +				default: { +					fprintf(stdout, "            Unknown EncodingVersion\n"); +				} +			} +		} +	} + +	return 0; +} diff --git a/test/libucsi/Makefile b/test/libucsi/Makefile new file mode 100644 index 0000000..1a831d0 --- /dev/null +++ b/test/libucsi/Makefile @@ -0,0 +1,13 @@ +# Makefile for linuxtv.org dvb-apps/test/libucsi + +binaries = testucsi + +CPPFLAGS += -I../../lib +LDLIBS   += ../../lib/libdvbapi/libdvbapi.a ../../lib/libdvbcfg/libdvbcfg.a \ +	    ../../lib/libdvbsec/libdvbsec.a  ../../lib/libucsi/libucsi.a + +.PHONY: all + +all: $(binaries) + +include ../../Make.rules diff --git a/test/libucsi/testucsi.c b/test/libucsi/testucsi.c new file mode 100644 index 0000000..e0a16c1 --- /dev/null +++ b/test/libucsi/testucsi.c @@ -0,0 +1,3528 @@ +/* + * section and descriptor parser test/sample application. + * + * Copyright (C) 2005 Andrew de Quincey (adq_dvb@lidskialf.net) + * + * This library 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 library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA + */ + +#include <libucsi/mpeg/descriptor.h> +#include <libucsi/mpeg/section.h> +#include <libucsi/dvb/descriptor.h> +#include <libucsi/dvb/section.h> +#include <libucsi/atsc/descriptor.h> +#include <libucsi/atsc/section.h> +#include <libucsi/transport_packet.h> +#include <libucsi/section_buf.h> +#include <libucsi/dvb/types.h> +#include <libdvbapi/dvbdemux.h> +#include <libdvbapi/dvbfe.h> +#include <libdvbcfg/dvbcfg_zapchannel.h> +#include <libdvbsec/dvbsec_api.h> +#include <libdvbsec/dvbsec_cfg.h> +#include <stdio.h> +#include <unistd.h> +#include <errno.h> +#include <stdarg.h> +#include <fcntl.h> + +void receive_data(int dvrfd, int timeout, int data_type); +void parse_section(uint8_t *buf, int len, int pid, int data_type); +void parse_dvb_section(uint8_t *buf, int len, int pid, int data_type, struct section *section); +void parse_atsc_section(uint8_t *buf, int len, int pid, int data_type, struct section *section); +void parse_descriptor(struct descriptor *d, int indent, int data_type); +void parse_dvb_descriptor(struct descriptor *d, int indent, int data_type); +void parse_atsc_descriptor(struct descriptor *d, int indent, int data_type); +void iprintf(int indent, char *fmt, ...); +void hexdump(int indent, char *prefix, uint8_t *buf, int buflen); +void atsctextdump(char *header, int indent, struct atsc_text *atext, int len); +int channels_cb(struct dvbcfg_zapchannel *channel, void *private); +void ts_from_file(char *filename, int data_type); + +#define TIME_CHECK_VAL 1131835761 +#define DURATION_CHECK_VAL 5643 + +#define MAX_TUNE_TIME 3000 +#define MAX_DUMP_TIME 60 + +#define DATA_TYPE_MPEG 0 +#define DATA_TYPE_DVB 1 +#define DATA_TYPE_ATSC 2 + + +struct dvbfe_handle *fe; +struct dvbfe_info feinfo; +int demuxfd; +int dvrfd; + +int main(int argc, char *argv[]) +{ +	int adapter; +	char *channelsfile; +	int pidlimit = -1; +	dvbdate_t dvbdate; +	dvbduration_t dvbduration; + +	// process arguments +	if ((argc < 3) || (argc > 4)) { +		fprintf(stderr, "Syntax: testucsi <adapter id>|-atscfile <filename> <zapchannels file> [<pid to limit to>]\n"); +		exit(1); +	} +	if (!strcmp(argv[1], "-atscfile")) { +		ts_from_file(argv[2], DATA_TYPE_ATSC); +		exit(0); +	} +	adapter = atoi(argv[1]); +	channelsfile = argv[2]; +	if (argc == 4) +		sscanf(argv[3], "%i", &pidlimit); +	printf("Using adapter %i\n", adapter); + +	// check the dvbdate conversion functions +	unixtime_to_dvbdate(TIME_CHECK_VAL, dvbdate); +	if (dvbdate_to_unixtime(dvbdate) != TIME_CHECK_VAL) { +		fprintf(stderr, "XXXX dvbdate function check failed (%i!=%i)\n", +			TIME_CHECK_VAL, (int) dvbdate_to_unixtime(dvbdate)); +		exit(1); +	} +	seconds_to_dvbduration(DURATION_CHECK_VAL, dvbduration); +	if (dvbduration_to_seconds(dvbduration) != DURATION_CHECK_VAL) { +		fprintf(stderr, "XXXX dvbduration function check failed (%i!=%i)\n", +			DURATION_CHECK_VAL, (int) dvbduration_to_seconds(dvbduration)); +		exit(1); +	} + +	// open the frontend +	if ((fe = dvbfe_open(adapter, 0, 0)) == NULL) { +		perror("open frontend"); +		exit(1); +	} +	dvbfe_get_info(fe, 0, &feinfo, DVBFE_INFO_QUERYTYPE_IMMEDIATE, 0); +	int data_type = DATA_TYPE_MPEG; +	switch(feinfo.type) { +	case DVBFE_TYPE_DVBS: +	case DVBFE_TYPE_DVBC: +	case DVBFE_TYPE_DVBT: +		data_type = DATA_TYPE_DVB; +		break; + +	case DVBFE_TYPE_ATSC: +		data_type = DATA_TYPE_ATSC; +		break; +	} + +	// open demux devices +	if ((demuxfd = dvbdemux_open_demux(adapter, 0, 0)) < 0) { +		perror("demux"); +		exit(1); +	} +	if ((dvrfd = dvbdemux_open_dvr(adapter, 0, 1, 1)) < 0) { +		perror("dvr"); +		exit(1); +	} + +	// make the demux buffer a bit larger +	if (dvbdemux_set_buffer(demuxfd, 1024*1024)) { +		perror("set buffer"); +		exit(1); +	} + +	// setup filter to capture stuff +	if (dvbdemux_set_pid_filter(demuxfd, pidlimit, DVBDEMUX_INPUT_FRONTEND, DVBDEMUX_OUTPUT_DVR, 1)) { +		perror("set pid filter"); +		exit(1); +	} + +	// process all the channels +	FILE *channels = fopen(channelsfile, "r"); +	if (channels == NULL) { +		fprintf(stderr, "Unable to open %s\n", channelsfile); +		exit(1); +	} +	dvbcfg_zapchannel_parse(channels, channels_cb, (void*) (long) data_type); +        return 0; +} + +void ts_from_file(char *filename, int data_type) { +	int fd = open(filename, O_RDONLY); +	if (fd < 0) { +		fprintf(stderr, "Unable to open file %s\n", filename); +		exit(1); +	} +	receive_data(fd, 1000000000, data_type); +} + +int channels_cb(struct dvbcfg_zapchannel *channel, void *private) +{ +	long data_type = (long) private; +	struct dvbsec_config sec; + +	if (dvbsec_cfg_find(NULL, "UNIVERSAL", &sec)) { +		fprintf(stderr, "Unable to find SEC id\n"); +		exit(1); +	} + +	if (dvbsec_set(fe, +			&sec, +			channel->polarization, +			DISEQC_SWITCH_UNCHANGED, +			DISEQC_SWITCH_UNCHANGED, +			&channel->fe_params, +			MAX_TUNE_TIME)) { +                fprintf(stderr, "Failed to lock!\n"); +        } else { +                printf("Tuned successfully!\n"); +                receive_data(dvrfd, MAX_DUMP_TIME, data_type); +        } + +	return 0; +} + +void receive_data(int _dvrfd, int timeout, int data_type) +{ +	unsigned char databuf[TRANSPORT_PACKET_LENGTH*20]; +	int sz; +	int pid; +	int i; +	int used; +	int section_status; +	time_t starttime; +	unsigned char continuities[TRANSPORT_MAX_PIDS]; +	struct section_buf *section_bufs[TRANSPORT_MAX_PIDS]; +	struct transport_packet *tspkt; +	struct transport_values tsvals; + +	// process the data +	starttime = time(NULL); +	memset(continuities, 0, sizeof(continuities)); +	memset(section_bufs, 0, sizeof(section_bufs)); +	while((time(NULL) - starttime) < timeout) { +		// got some! +		if ((sz = read(_dvrfd, databuf, sizeof(databuf))) < 0) { +			if (errno == EOVERFLOW) { +				fprintf(stderr, "data overflow!\n"); +				continue; +			} else if (errno == EAGAIN) { +				usleep(100); +				continue; +			} else { +				perror("read error"); +				exit(1); +			} +		} +		for(i=0; i < sz; i+=TRANSPORT_PACKET_LENGTH) { +			// parse the transport packet +			tspkt = transport_packet_init(databuf + i); +			if (tspkt == NULL) { +				fprintf(stderr, "XXXX Bad sync byte\n"); +				continue; +			} +			pid = transport_packet_pid(tspkt); + +			// extract all TS packet values even though we don't need them (to check for +			// library segfaults etc) +			if (transport_packet_values_extract(tspkt, &tsvals, 0xffff) < 0) { +				fprintf(stderr, "XXXX Bad packet received (pid:%04x)\n", pid); +				continue; +			} + +			// check continuity +			if (transport_packet_continuity_check(tspkt, +			    tsvals.flags & transport_adaptation_flag_discontinuity, +			    continuities + pid)) { +				fprintf(stderr, "XXXX Continuity error (pid:%04x)\n", pid); +				continuities[pid] = 0; +				if (section_bufs[pid] != NULL) { +					section_buf_reset(section_bufs[pid]); +				} +				continue; +			} + +			// allocate section buf if we don't have one already +			if (section_bufs[pid] == NULL) { +				section_bufs[pid] = (struct section_buf*) +					malloc(sizeof(struct section_buf) + DVB_MAX_SECTION_BYTES); +				if (section_bufs[pid] == NULL) { +					fprintf(stderr, "Failed to allocate section buf (pid:%04x)\n", pid); +					exit(1); +				} +				section_buf_init(section_bufs[pid], DVB_MAX_SECTION_BYTES); +			} + +			// process the payload data as a section +			while(tsvals.payload_length) { +				used = section_buf_add_transport_payload(section_bufs[pid], +									 tsvals.payload, +									 tsvals.payload_length, +									 tspkt->payload_unit_start_indicator, +									 §ion_status); +				tspkt->payload_unit_start_indicator = 0; +				tsvals.payload_length -= used; +				tsvals.payload += used; + +				if (section_status == 1) { +					parse_section(section_buf_data(section_bufs[pid]), +						      section_bufs[pid]->len, pid, data_type); +					section_buf_reset(section_bufs[pid]); +				} else if (section_status < 0) { +					// some kind of error - just discard +					fprintf(stderr, "XXXX bad section %04x %i\n",pid, section_status); +					section_buf_reset(section_bufs[pid]); +				} +			} +		} +	} +} + +void parse_section(uint8_t *buf, int len, int pid, int data_type) +{ +	struct section *section; +	struct section_ext *section_ext = NULL; + +	if ((section = section_codec(buf, len)) == NULL) { +		return; +	} + +	switch(section->table_id) { +	case stag_mpeg_program_association: +	{ +		struct mpeg_pat_section *pat; +		struct mpeg_pat_program *cur; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode PAT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((pat = mpeg_pat_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX PAT section decode error\n"); +			return; +		} +		printf("SCT transport_stream_id:0x%04x\n", mpeg_pat_section_transport_stream_id(pat)); +		mpeg_pat_section_programs_for_each(pat, cur) { +			printf("\tSCT program_number:0x%04x pid:0x%04x\n", cur->program_number, cur->pid); +		} +		break; +	} + +	case stag_mpeg_conditional_access: +	{ +		struct mpeg_cat_section *cat; +		struct descriptor *curd; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode CAT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((cat = mpeg_cat_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX CAT section decode error\n"); +			return; +		} +		mpeg_cat_section_descriptors_for_each(cat, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		break; +	} + +	case stag_mpeg_program_map: +	{ +		struct mpeg_pmt_section *pmt; +		struct descriptor *curd; +		struct mpeg_pmt_stream *cur_stream; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode PMT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((pmt = mpeg_pmt_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX PMT section decode error\n"); +			return; +		} +		printf("SCT program_number:0x%04x pcr_pid:0x%02x\n", mpeg_pmt_section_program_number(pmt), pmt->pcr_pid); +		mpeg_pmt_section_descriptors_for_each(pmt, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		mpeg_pmt_section_streams_for_each(pmt, cur_stream) { +			printf("\tSCT stream_type:0x%02x pid:0x%04x\n", cur_stream->stream_type, cur_stream->pid); +			mpeg_pmt_stream_descriptors_for_each(cur_stream, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		break; +	} + +	case stag_mpeg_transport_stream_description: +	{ +		struct mpeg_tsdt_section *tsdt; +		struct descriptor *curd; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode TSDT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((tsdt = mpeg_tsdt_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX TSDT section decode error\n"); +			return; +		} +		mpeg_tsdt_section_descriptors_for_each(tsdt, curd) { +			parse_descriptor(curd, 1, data_type); +		} + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	case stag_mpeg_metadata: +	{ +		struct mpeg_metadata_section *metadata; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode metadata (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((metadata = mpeg_metadata_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX metadata section decode error\n"); +			return; +		} +		printf("SCT random_access_indicator:%i decoder_config_flag:%i fragment_indicator:%i service_id:%02x\n", +		       mpeg_metadata_section_random_access_indicator(metadata), +		       mpeg_metadata_section_decoder_config_flag(metadata), +		       mpeg_metadata_section_fragment_indicator(metadata), +		       mpeg_metadata_section_service_id(metadata)); +		hexdump(0, "SCT ", mpeg_metadata_section_data(metadata), mpeg_metadata_section_data_length(metadata)); + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	case stag_mpeg_iso14496_scene_description: +	case stag_mpeg_iso14496_object_description: +	{ +		struct mpeg_odsmt_section *odsmt; +		struct mpeg_odsmt_stream *cur_stream; +		struct descriptor *curd; +		int _index; +		uint8_t *objects; +		size_t objects_length; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode ISO14496 (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((odsmt = mpeg_odsmt_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "XXXX ISO14496 section decode error\n"); +			return; +		} +		printf("SCT PID:0x%04x\n", mpeg_odsmt_section_pid(odsmt)); +		mpeg_odsmt_section_streams_for_each(osdmt, cur_stream, _index) { +			if (odsmt->stream_count == 0) { +				printf("\tSCT SINGLE 0x%04x\n", cur_stream->u.single.esid); +			} else { +				printf("\tSCT MULTI 0x%04x 0x%02x\n", cur_stream->u.multi.esid, cur_stream->u.multi.fmc); +			} +			mpeg_odsmt_stream_descriptors_for_each(osdmt, cur_stream, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		objects = mpeg_odsmt_section_object_descriptors(odsmt, &objects_length); +		if (objects == NULL) { +			printf("SCT XXXX OSDMT parse error\n"); +			break; +		} +		hexdump(1, "SCT ", objects, objects_length); + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	default: +		switch(data_type) { +		case DATA_TYPE_DVB: +			parse_dvb_section(buf, len, pid, data_type, section); +			break; + +		case DATA_TYPE_ATSC: +			parse_atsc_section(buf, len, pid, data_type, section); +			break; + +		default: +			fprintf(stderr, "SCT XXXX Unknown table_id:0x%02x (pid:0x%04x)\n", +				section->table_id, pid); +//			hexdump(0, "SCT ", buf, len); +			return; +		} +	} + +	printf("\n"); +} + +void parse_dvb_section(uint8_t *buf, int len, int pid, int data_type, struct section *section) +{ +	struct section_ext *section_ext = NULL; + +	switch(section->table_id) { +	case stag_dvb_network_information_actual: +	case stag_dvb_network_information_other: +	{ +		struct dvb_nit_section *nit; +	   	struct descriptor *curd; +		struct dvb_nit_section_part2 *part2; +		struct dvb_nit_transport *cur_transport; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode NIT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((nit = dvb_nit_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX NIT section decode error\n"); +			return; +		} +		printf("SCT network_id:0x%04x\n", dvb_nit_section_network_id(nit)); +		dvb_nit_section_descriptors_for_each(nit, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		part2 = dvb_nit_section_part2(nit); +		dvb_nit_section_transports_for_each(nit, part2, cur_transport) { +			printf("\tSCT transport_stream_id:0x%04x original_network_id:0x%04x\n", cur_transport->transport_stream_id, cur_transport->original_network_id); +			dvb_nit_transport_descriptors_for_each(cur_transport, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		break; +	} + +	case stag_dvb_service_description_actual: +	case stag_dvb_service_description_other: +	{ +		struct dvb_sdt_section *sdt; +		struct dvb_sdt_service *cur_service; +		struct descriptor *curd; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode SDT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((sdt = dvb_sdt_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "XXXX SDT section decode error\n"); +			return; +		} +		printf("SCT transport_stream_id:0x%04x original_network_id:0x%04x\n", dvb_sdt_section_transport_stream_id(sdt), sdt->original_network_id); +		dvb_sdt_section_services_for_each(sdt, cur_service) { +			printf("\tSCT service_id:0x%04x eit_schedule_flag:%i eit_present_following_flag:%i running_status:%i free_ca_mode:%i\n", +			       cur_service->service_id, +			       cur_service->eit_schedule_flag, +			       cur_service->eit_present_following_flag, +			       cur_service->running_status, +			       cur_service->free_ca_mode); +			dvb_sdt_service_descriptors_for_each(cur_service, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		break; +	} + +	case stag_dvb_bouquet_association: +	{ +		struct dvb_bat_section *bat; +		struct descriptor *curd; +		struct dvb_bat_section_part2 *part2; +		struct dvb_bat_transport *cur_transport; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode BAT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((bat = dvb_bat_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX BAT section decode error\n"); +			return; +		} +		printf("SCT bouquet_id:0x%04x\n", dvb_bat_section_bouquet_id(bat)); +		dvb_bat_section_descriptors_for_each(bat, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		part2 = dvb_bat_section_part2(bat); +		dvb_bat_section_transports_for_each(part2, cur_transport) { +			printf("\tSCT transport_stream_id:0x%04x original_network_id:0x%04x\n", +			       cur_transport->transport_stream_id, +			       cur_transport->original_network_id); +			dvb_bat_transport_descriptors_for_each(cur_transport, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		break; +	} + +	case stag_dvb_update_notification: +	case stag_dvb_ip_mac_notification: +	{ +		struct dvb_int_section *_int; +		struct descriptor *curd; +		struct dvb_int_target *cur_target; +		struct dvb_int_operational_loop *operational_loop; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode INT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((_int = dvb_int_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "XXXX INT section decode error\n"); +			return; +		} +		printf("SCT action_type:0x%02x platform_id_hash:0x%02x platform_id:0x%06x processing_order:0x%02x\n", +		       dvb_int_section_action_type(_int), +		       dvb_int_section_platform_id_hash(_int), +		       _int->platform_id, +		       _int->processing_order); +		dvb_int_section_platform_descriptors_for_each(_int, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		dvb_int_section_target_loop_for_each(_int, cur_target) { +			dvb_int_target_target_descriptors_for_each(cur_target, curd) { +				parse_descriptor(curd, 2, data_type); +			} +			operational_loop = dvb_int_target_operational_loop(cur_target); +			dvb_int_operational_loop_operational_descriptors_for_each(operational_loop, curd) { +				parse_descriptor(curd, 3, data_type); +			} +		} + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	case stag_dvb_event_information_nownext_actual: +	case stag_dvb_event_information_nownext_other: +	case 0x50: case 0x51: case 0x52: case 0x53: case 0x54: case 0x55: case 0x56: case 0x57: +	case 0x58: case 0x59: case 0x5a: case 0x5b: case 0x5c: case 0x5d: case 0x5e: case 0x5f: +	case 0x60: case 0x61: case 0x62: case 0x63: case 0x64: case 0x65: case 0x66: case 0x67: +	case 0x68: case 0x69: case 0x6a: case 0x6b: case 0x6c: case 0x6d: case 0x6e: case 0x6f: +	{ +		struct dvb_eit_section *eit; +		struct dvb_eit_event *cur_event; +		struct descriptor *curd; +		time_t start_time; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode EIT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((eit = dvb_eit_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "XXXX EIT section decode error\n"); +			return; +		} +		printf("SCT service_id:0x%04x transport_stream_id:0x%04x original_network_id:0x%04x segment_last_section_number:0x%02x last_table_id:0x%02x\n", +		       dvb_eit_section_service_id(eit), +		       eit->transport_stream_id, +		       eit->original_network_id, +		       eit->segment_last_section_number, +		       eit->last_table_id); +		dvb_eit_section_events_for_each(eit, cur_event) { +			start_time = dvbdate_to_unixtime(cur_event->start_time); +			printf("\tSCT event_id:0x%04x duration:%i running_status:%i free_ca_mode:%i start_time:%i -- %s", +			       cur_event->event_id, +			       dvbduration_to_seconds(cur_event->duration), +			       cur_event->running_status, +			       cur_event->free_ca_mode, +			       (int) start_time, +			       ctime(&start_time)); +			dvb_eit_event_descriptors_for_each(cur_event, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		break; +	} + +	case stag_dvb_time_date: +	{ +		struct dvb_tdt_section *tdt; +		time_t dvbtime; + +		printf("SCT Decode TDT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((tdt = dvb_tdt_section_codec(section)) == NULL) { +			fprintf(stderr, "XXXX TDT section decode error\n"); +			return; +		} +		dvbtime = dvbdate_to_unixtime(tdt->utc_time); +		printf("SCT Time: %i -- %s", (int) dvbtime, ctime(&dvbtime)); +		break; +	} + +	case stag_dvb_running_status: +	{ +		struct dvb_rst_section *rst; +		struct dvb_rst_status *cur_status; + +		printf("SCT Decode RST (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((rst = dvb_rst_section_codec(section)) == NULL) { +			fprintf(stderr, "SCT XXXX RST section decode error\n"); +			return; +		} +		dvb_rst_section_statuses_for_each(rst, cur_status) { +			printf("\tSCT transport_stream_id:0x%04x original_network_id:0x%04x service_id:0x%04x event_id:0x%04x running_status:%i\n", +			       cur_status->transport_stream_id, +			       cur_status->original_network_id, +			       cur_status->service_id, +			       cur_status->event_id, +			       cur_status->running_status); +		} + +//		hexdump(0, "SCT ", buf, len); +//		getchar(); +		break; +	} + +	case stag_dvb_stuffing: +	{ +		struct dvb_st_section *st; + +		printf("SCT Decode ST (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((st = dvb_st_section_codec(section)) == NULL) { +			fprintf(stderr, "SCT XXXX ST section decode error\n"); +			return; +		} +		printf("SCT Length: %i\n", dvb_st_section_data_length(st)); +		break; +	} + +	case stag_dvb_time_offset: +	{ +		struct dvb_tot_section *tot; +		struct descriptor *curd; +		time_t dvbtime; + +		if (section_check_crc(section)) +			return; +		printf("SCT Decode TOT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((tot = dvb_tot_section_codec(section)) == NULL) { +			fprintf(stderr, "SCT XXXX TOT section decode error\n"); +			return; +		} +		dvbtime = dvbdate_to_unixtime(tot->utc_time); +		printf("SCT utc_time: %i -- %s", (int) dvbtime, ctime(&dvbtime)); +		dvb_tot_section_descriptors_for_each(tot, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		break; +	} + +	case stag_dvb_tva_container: +	{ +		struct dvb_tva_container_section *tva; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode tva (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((tva = dvb_tva_container_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX tva section decode error\n"); +			return; +		} +		printf("SCT container_id:%04x\n", +		       dvb_tva_container_section_container_id(tva)); +		hexdump(0, "SCT ", dvb_tva_container_section_data(tva), dvb_tva_container_section_data_length(tva)); + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	case stag_dvb_discontinuity_information: +	{ +		struct dvb_dit_section *dit; + +		printf("SCT Decode DIT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((dit = dvb_dit_section_codec(section)) == NULL) { +			fprintf(stderr, "SCT XXXX DIT section decode error\n"); +			return; +		} +		printf("SCT transition_flag:%i\n", dit->transition_flag); + +//		hexdump(0, "SCT ", buf, len); +//		getchar(); +		break; +	} + +	case stag_dvb_selection_information: +	{ +		struct dvb_sit_section *sit; +		struct descriptor *curd; +		struct dvb_sit_service *cur_service; + +		if ((section_ext = section_ext_decode(section, 1)) == NULL) { +			return; +		} +		printf("SCT Decode SIT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((sit = dvb_sit_section_codec(section_ext)) == NULL) { +			fprintf(stderr, "SCT XXXX SIT section decode error\n"); +			return; +		} +		dvb_sit_section_descriptors_for_each(sit, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		dvb_sit_section_services_for_each(sit, cur_service) { +			printf("\tSCT service_id:0x%04x running_status:%i\n", cur_service->service_id, cur_service->running_status); +			dvb_sit_service_descriptors_for_each(cur_service, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	default: +		fprintf(stderr, "SCT XXXX Unknown table_id:0x%02x (pid:0x%04x)\n", section->table_id, pid); +//		hexdump(0, "SCT ", buf, len); +		return; +	} +} + +void parse_atsc_section(uint8_t *buf, int len, int pid, int data_type, struct section *section) +{ +	struct section_ext *section_ext = NULL; +	struct atsc_section_psip *section_psip = NULL; +	if ((section_ext = section_ext_decode(section, 1)) == NULL) { +		return; +	} +	if ((section_psip = atsc_section_psip_decode(section_ext)) == NULL) { +		return; +	} + +	printf("SCT protocol_version:%i\n", section_psip->protocol_version); + +	switch(section->table_id) { +	case stag_atsc_master_guide: +	{ +		struct atsc_mgt_section *mgt; +		struct atsc_mgt_table *cur_table; +		struct atsc_mgt_section_part2 *part2; +		struct descriptor *curd; +		int idx; + +		printf("SCT Decode MGT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((mgt = atsc_mgt_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX MGT section decode error\n"); +			return; +		} +		atsc_mgt_section_tables_for_each(mgt, cur_table, idx) { +			printf("\tSCT table_type:0x%04x table_type_PID:%04x table_type_version_number:%i number_bytes:%i\n", +			       cur_table->table_type, +			       cur_table->table_type_PID, +			       cur_table->table_type_version_number, +			       cur_table->number_bytes); +			atsc_mgt_table_descriptors_for_each(cur_table, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} + +		part2 = atsc_mgt_section_part2(mgt); +		atsc_mgt_section_part2_descriptors_for_each(part2, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		break; +	} + +	case stag_atsc_terrestrial_virtual_channel: +	{ +		struct atsc_tvct_section *tvct; +		struct atsc_tvct_channel *cur_channel; +		struct atsc_tvct_section_part2 *part2; +		struct descriptor *curd; +		int idx; + +		printf("SCT Decode TVCT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((tvct = atsc_tvct_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX TVCT section decode error\n"); +			return; +		} +		printf("\tSCT tranport_stream_id:0x%04x\n", +		       atsc_tvct_section_transport_stream_id(tvct)); + +		atsc_tvct_section_channels_for_each(tvct, cur_channel, idx) { +			hexdump(0, "SCT short_name ", (uint8_t*) cur_channel->short_name, 7*2); + +			printf("\tSCT major_channel_number:%04x minor_channel_number:%04x modulation_mode:%02x carrier_frequency:%i channel_TSID:%04x program_number:%04x ETM_location:%i access_controlled:%i hidden:%i hide_guide:%i service_type:%02x source_id:%04x\n", +			       cur_channel->major_channel_number, +			       cur_channel->minor_channel_number, +			       cur_channel->modulation_mode, +			       cur_channel->carrier_frequency, +			       cur_channel->channel_TSID, +			       cur_channel->program_number, +			       cur_channel->ETM_location, +			       cur_channel->access_controlled, +			       cur_channel->hidden, +			       cur_channel->hide_guide, +			       cur_channel->service_type, +			       cur_channel->source_id); +			atsc_tvct_channel_descriptors_for_each(cur_channel, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} + +		part2 = atsc_tvct_section_part2(tvct); +		atsc_tvct_section_part2_descriptors_for_each(part2, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		break; +	} + +	case stag_atsc_cable_virtual_channel: +	{ +		struct atsc_cvct_section *cvct; +		struct atsc_cvct_channel *cur_channel; +		struct atsc_cvct_section_part2 *part2; +		struct descriptor *curd; +		int idx; + +		printf("SCT Decode CVCT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((cvct = atsc_cvct_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX CVCT section decode error\n"); +			return; +		} +		printf("\tSCT tranport_stream_id:0x%04x\n", +		       atsc_cvct_section_transport_stream_id(cvct)); + +		atsc_cvct_section_channels_for_each(cvct, cur_channel, idx) { +			hexdump(0, "SCT short_name ", (uint8_t*) cur_channel->short_name, 7*2); + +			printf("\tSCT major_channel_number:%04x minor_channel_number:%04x modulation_mode:%02x carrier_frequency:%i channel_TSID:%04x program_number:%04x ETM_location:%i access_controlled:%i hidden:%i path_select:%i out_of_band:%i hide_guide:%i service_type:%02x source_id:%04x\n", +			       cur_channel->major_channel_number, +			       cur_channel->minor_channel_number, +			       cur_channel->modulation_mode, +			       cur_channel->carrier_frequency, +			       cur_channel->channel_TSID, +			       cur_channel->program_number, +			       cur_channel->ETM_location, +			       cur_channel->access_controlled, +			       cur_channel->hidden, +			       cur_channel->path_select, +			       cur_channel->out_of_band, +			       cur_channel->hide_guide, +			       cur_channel->service_type, +			       cur_channel->source_id); +			atsc_cvct_channel_descriptors_for_each(cur_channel, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} + +		part2 = atsc_cvct_section_part2(cvct); +		atsc_cvct_section_part2_descriptors_for_each(part2, curd) { +			parse_descriptor(curd, 1, data_type); +		} +		break; +	} + +	case stag_atsc_rating_region: +	{ +		struct atsc_rrt_section *rrt; +		struct atsc_rrt_section_part2 *part2; +		struct atsc_rrt_dimension *cur_dimension; +		struct atsc_rrt_dimension_part2 *dpart2; +		struct atsc_rrt_dimension_value *cur_value; +		struct atsc_rrt_dimension_value_part2 *vpart2; +		struct atsc_rrt_section_part3 *part3; +		struct descriptor *curd; +		int didx; +		int vidx; + +		printf("SCT Decode RRT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((rrt = atsc_rrt_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX RRT section decode error\n"); +			return; +		} +		printf("\tSCT rating_region:0x%02x\n", +		       atsc_rrt_section_rating_region(rrt)); +		atsctextdump("SCT region_name:", 1, +			     atsc_rrt_section_rating_region_name_text(rrt), +			     rrt->rating_region_name_length); + +		part2 = atsc_rrt_section_part2(rrt); +		atsc_rrt_section_dimensions_for_each(part2, cur_dimension, didx) { +			atsctextdump("SCT dimension_name:", 2, +				     atsc_rrt_dimension_name_text(cur_dimension), +				     cur_dimension->dimension_name_length); + + 			dpart2 = atsc_rrt_dimension_part2(cur_dimension); +			printf("\tSCT graduated_scale:%i\n", +			       dpart2->graduated_scale); + +			atsc_rrt_dimension_part2_values_for_each(dpart2, cur_value, vidx) { +				atsctextdump("SCT value_abbrev_name:", 3, +					     atsc_rrt_dimension_value_abbrev_rating_value_text(cur_value), +					     cur_value->abbrev_rating_value_length); + +				vpart2 = atsc_rrt_dimension_value_part2(cur_value); +				atsctextdump("SCT value_text:", 3, +					     atsc_rrt_dimension_value_part2_rating_value_text(vpart2), +					     vpart2->rating_value_length); +			} +		} + +		part3 = atsc_rrt_section_part3(part2); +		atsc_rrt_section_part3_descriptors_for_each(part3, curd) { +			parse_descriptor(curd, 1, data_type); +		} + +		hexdump(0, "SCT ", buf, len); +		getchar(); +		break; +	} + +	case stag_atsc_event_information: +	{ +		struct atsc_eit_section *eit; +		struct atsc_eit_event *cur_event; +		struct atsc_eit_event_part2 *part2; +		struct descriptor *curd; +		int idx; + +		printf("SCT Decode EIT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((eit = atsc_eit_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX EIT section decode error\n"); +			return; +		} +		printf("\tSCT source_id:0x%04x\n", +		       atsc_eit_section_source_id(eit)); + +		atsc_eit_section_events_for_each(eit, cur_event, idx) { +			printf("\t\tSCT event_id:%04x start_time:%i ETM_location:%i length_in_secs:%i\n", +			       cur_event->event_id, +			       cur_event->start_time, +			       cur_event->ETM_location, +			       cur_event->length_in_seconds); + +			atsctextdump("SCT title:", 2, +				     atsc_eit_event_name_title_text(cur_event), +				     cur_event->title_length); + +			part2 = atsc_eit_event_part2(cur_event); + +			atsc_eit_event_part2_descriptors_for_each(part2, curd) { +				parse_descriptor(curd, 2, data_type); +			} +		} +		break; +	} + +	case stag_atsc_extended_text: +	{ +		struct atsc_ett_section *ett; + +		printf("SCT Decode ETT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((ett = atsc_ett_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX ETT section decode error\n"); +			return; +		} +		printf("\tSCT ETM_source_id:0x%04x ETM_sub_id:%04x ETM_type:%02x\n", +		       ett->ETM_source_id, +		       ett->ETM_sub_id, +		       ett->ETM_type); +		atsctextdump("SCT text:", 1, +			     atsc_ett_section_extended_text_message(ett), +			     atsc_ett_section_extended_text_message_length(ett)); +		break; +	} + +	case stag_atsc_system_time: +	{ +		struct atsc_stt_section *stt; +		struct descriptor *curd; + +		printf("SCT Decode STT (pid:0x%04x) (table:0x%02x)\n", pid, section->table_id); +		if ((stt = atsc_stt_section_codec(section_psip)) == NULL) { +			fprintf(stderr, "SCT XXXX STT section decode error\n"); +			return; +		} +		printf("\tSCT system_time:%i gps_utc_offset:%i DS_status:%i DS_day_of_month:%i DS_hour:%i\n", +		       stt->system_time, +		       stt->gps_utc_offset, +		       stt->DS_status, +		       stt->DS_day_of_month, +		       stt->DS_hour); +		atsc_stt_section_descriptors_for_each(stt, curd) { +			parse_descriptor(curd, 2, data_type); +		} +		break; +	} + +	default: +		fprintf(stderr, "SCT XXXX Unknown table_id:0x%02x (pid:0x%04x)\n", section->table_id, pid); +		hexdump(0, "SCT ", buf, len); +		return; +	} +} + +void parse_descriptor(struct descriptor *d, int indent, int data_type) +{ +	switch(d->tag) { +	case dtag_mpeg_video_stream: +	{ +		struct mpeg_video_stream_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_video_stream_descriptor\n"); +		dx = mpeg_video_stream_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_video_stream_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC multiple_frame_rate_flag:%i frame_rate_code:%i mpeg_1_only_flag:%i constrained_parameter_flag:%i still_picture_flag:%i\n", +			dx->multiple_frame_rate_flag, +			dx->frame_rate_code, +			dx->mpeg_1_only_flag, +			dx->constrained_parameter_flag, +			dx->still_picture_flag); +		if (!dx->mpeg_1_only_flag) { +			struct mpeg_video_stream_extra *extra = mpeg_video_stream_descriptor_extra(dx); +			iprintf(indent, "DSC profile_and_level_indication:0x%02x chroma_format:%i frame_rate_extension:%i\n", +				extra->profile_and_level_indication, +				extra->chroma_format, +				extra->frame_rate_extension); +		} +		break; +	} + +	case dtag_mpeg_audio_stream: +	{ +		struct mpeg_audio_stream_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_audio_stream_descriptor\n"); +		dx = mpeg_audio_stream_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_audio_stream_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC free_format_flag:%i id:%i layer:%i variable_rate_audio_indicator:%i\n", +			dx->free_format_flag, +			dx->id, +			dx->layer, +			dx->variable_rate_audio_indicator); +		break; +	} + +	case dtag_mpeg_hierarchy: +	{ +		struct mpeg_hierarchy_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_hierarchy_descriptor\n"); +		dx = mpeg_hierarchy_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_hierarchy_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC hierarchy_type:%i hierarchy_layer_index:%i hierarchy_embedded_layer_index:%i hierarchy_channel:%i\n", +			dx->hierarchy_type, +			dx->hierarchy_layer_index, +			dx->hierarchy_embedded_layer_index, +			dx->hierarchy_channel); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_registration: +	{ +		struct mpeg_registration_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_registration_descriptor\n"); +		dx = mpeg_registration_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_registration_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC format_identifier:0x%x\n", +			dx->format_identifier); +		iprintf(indent, "DSC additional_id_info:\n"); +		hexdump(indent, "DSC ", +			mpeg_registration_descriptor_additional_id_info(dx), +			mpeg_registration_descriptor_additional_id_info_length(dx)); +		break; +	} + +	case dtag_mpeg_data_stream_alignment: +	{ +		struct mpeg_data_stream_alignment_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_data_stream_alignment_descriptor\n"); +		dx = mpeg_data_stream_alignment_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_data_stream_alignment_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC alignment_type:%i\n", +			dx->alignment_type); +		break; +	} + +	case dtag_mpeg_target_background_grid: +	{ +		struct mpeg_target_background_grid_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_target_background_grid_descriptor\n"); +		dx = mpeg_target_background_grid_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_target_background_grid_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC horizontal_size:%i vertical_size:%i aspect_ratio_information:%i\n", +			dx->horizontal_size, +		        dx->vertical_size, +		        dx->aspect_ratio_information); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_video_window: +	{ +		struct mpeg_video_window_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_video_window_descriptor\n"); +		dx = mpeg_video_window_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_video_window_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC horizontal_offset:%i vertical_offset:%i window_priority:%i\n", +			dx->horizontal_offset, +			dx->vertical_offset, +			dx->window_priority); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_ca: +	{ +		struct mpeg_ca_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_ca_descriptor\n"); +		dx = mpeg_ca_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_ca_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC ca_system_id:0x%04x ca_pid:0x%04x\n", +			dx->ca_system_id, +			dx->ca_pid); +		iprintf(indent, "DSC data:\n"); +		hexdump(indent, "DSC ", mpeg_ca_descriptor_data(dx), mpeg_ca_descriptor_data_length(dx)); +		break; +	} + +	case dtag_mpeg_iso_639_language: +	{ +		struct mpeg_iso_639_language_descriptor *dx; +		struct mpeg_iso_639_language_code *cur_lang; + +		iprintf(indent, "DSC Decode mpeg_iso_639_language_descriptor\n"); +		dx = mpeg_iso_639_language_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_iso_639_language_descriptor decode error\n"); +			return; +		} +		mpeg_iso_639_language_descriptor_languages_for_each(dx, cur_lang) { +			iprintf(indent+1, "DSC language_code:%.3s audio_type:0x%02x\n", +				cur_lang->language_code, +				cur_lang->audio_type); +		} +		break; +	} + +	case dtag_mpeg_system_clock: +	{ +		struct mpeg_system_clock_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_system_clock_descriptor\n"); +		dx = mpeg_system_clock_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_system_clock_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC external_clock_reference_indicator:%i clock_accuracy_integer:%i clock_accuracy_exponent:%i\n", +			dx->external_clock_reference_indicator, +			dx->clock_accuracy_integer, +		        dx->clock_accuracy_exponent); +		break; +	} + +	case dtag_mpeg_multiplex_buffer_utilization: +	{ +		struct mpeg_multiplex_buffer_utilization_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_multiplex_buffer_utilization_descriptor\n"); +		dx = mpeg_multiplex_buffer_utilization_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_multiplex_buffer_utilization_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC bound_valid_flag:%i ltw_offset_lower_bound:%i ltw_offset_upper_bound:%i\n", +			dx->bound_valid_flag, +			dx->ltw_offset_lower_bound, +			dx->ltw_offset_upper_bound); +		break; +	} + +	case dtag_mpeg_copyright: +	{ +		struct mpeg_copyright_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_copyright_descriptor\n"); +		dx = mpeg_copyright_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_copyright_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC copyright_identifier:0x%08x\n", +			dx->copyright_identifier); +		iprintf(indent, "DSC data:\n"); +		hexdump(indent, "DSC ", mpeg_copyright_descriptor_data(dx), mpeg_copyright_descriptor_data_length(dx)); +		break; +	} + +	case dtag_mpeg_maximum_bitrate: +	{ +		struct mpeg_maximum_bitrate_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_maximum_bitrate_descriptor\n"); +		dx = mpeg_maximum_bitrate_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_maximum_bitrate_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC maximum_bitrate:%i\n", +			dx->maximum_bitrate); +		break; +	} + +	case dtag_mpeg_private_data_indicator: +	{ +		struct mpeg_private_data_indicator_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_private_data_indicator_descriptor\n"); +		dx = mpeg_private_data_indicator_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_private_data_indicator_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC private_data_indicator:0x%x\n", +			dx->private_data_indicator); +		break; +	} + +	case dtag_mpeg_smoothing_buffer: +	{ +		struct mpeg_smoothing_buffer_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_smoothing_buffer_descriptor\n"); +		dx = mpeg_smoothing_buffer_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_smoothing_buffer_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC sb_leak_rate:%i sb_size:%i\n", +			dx->sb_leak_rate, +		        dx->sb_size); +		break; +	} + +	case dtag_mpeg_std: +	{ +		struct mpeg_std_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_std_descriptor\n"); +		dx = mpeg_std_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_std_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC leak_valid_flag:%i\n", +			dx->leak_valid_flag); +		break; +	} + +	case dtag_mpeg_ibp: +	{ +		struct mpeg_ibp_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_ibp_descriptor\n"); +		dx = mpeg_ibp_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_ibp_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC closed_gop_flag:%i identical_gop_flag:%i max_gop_length:%i\n", +			dx->closed_gop_flag, dx->identical_gop_flag, dx->max_gop_length); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_4_video: +	{ +		struct mpeg4_video_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg4_video_descriptor\n"); +		dx = mpeg4_video_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg4_video_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC mpeg4_visual_profile_and_level:0x%02x\n", +			dx->mpeg4_visual_profile_and_level); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_4_audio: +	{ +		struct mpeg4_audio_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg4_audio_descriptor\n"); +		dx = mpeg4_audio_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg4_audio_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC mpeg4_audio_profile_and_level:0x%02x\n", +			dx->mpeg4_audio_profile_and_level); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_iod: +	{ +		struct mpeg_iod_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_iod_descriptor\n"); +		dx = mpeg_iod_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_iod_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC scope_of_iod_label:0x%08x iod_label:0x%02x\n", +			dx->scope_of_iod_label, dx->iod_label); +		iprintf(indent, "DSC iod:\n"); +		hexdump(indent, "DSC ", mpeg_iod_descriptor_iod(dx), mpeg_iod_descriptor_iod_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_sl: +	{ +		struct mpeg_sl_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_sl_descriptor\n"); +		dx = mpeg_sl_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_sl_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC es_id:0x%04x\n", +			dx->es_id); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_fmc: +	{ +		struct mpeg_fmc_descriptor *dx; +		struct mpeg_flex_mux *cur_fm; + +		iprintf(indent, "DSC Decode mpeg_fmc_descriptor\n"); +		dx = mpeg_fmc_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_fmc_descriptor_descriptor decode error\n"); +			return; +		} +		mpeg_fmc_descriptor_muxes_for_each(dx, cur_fm) { +			iprintf(indent+1, "DSC es_id:0x%04x flex_mux_channel:0x%02x\n", +				cur_fm->es_id, +				cur_fm->flex_mux_channel); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_external_es_id: +	{ +		struct mpeg_external_es_id_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_external_es_id_descriptor\n"); +		dx = mpeg_external_es_id_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_external_es_id_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC external_es_id:0x%04x\n", +			dx->external_es_id); +		break; +	} + +	case dtag_mpeg_muxcode: +	{ +		struct mpeg_muxcode_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_muxcode_descriptor\n"); +		dx = mpeg_muxcode_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_muxcode_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC entries:\n"); +		hexdump(indent, "DSC ", mpeg_muxcode_descriptor_entries(dx), mpeg_muxcode_descriptor_entries_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_fmxbuffer_size: +	{ +		struct mpeg_fmxbuffer_size_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_fmxbuffer_size_descriptor\n"); +		dx = mpeg_fmxbuffer_size_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_fmxbuffer_size_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC descriptors:\n"); +		hexdump(indent, "DSC ", mpeg_fmxbuffer_size_descriptor_descriptors(dx), mpeg_fmxbuffer_size_descriptor_descriptors_length(dx)); +		break; +	} + +	case dtag_mpeg_multiplex_buffer: +	{ +		struct mpeg_multiplex_buffer_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_multiplex_buffer_descriptor\n"); +		dx = mpeg_multiplex_buffer_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_multiplex_buffer_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC mb_buffer_size:%i tb_leak_rate:%i\n", +			dx->mb_buffer_size, dx->tb_leak_rate); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_content_labelling: +	{ +		struct mpeg_content_labelling_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_content_labelling_descriptor\n"); +		dx = mpeg_content_labelling_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_content_labelling_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC metadata_application_format:%04x\n", +			dx->metadata_application_format); +		struct mpeg_content_labelling_descriptor_application_format_identifier *id = +			mpeg_content_labelling_descriptor_id(dx); +		if (id != NULL) { +			iprintf(indent, "DSC application_format_id:%04x\n", +				id->id); +		} +		struct mpeg_content_labelling_descriptor_flags *flags = +			mpeg_content_labelling_descriptor_flags(dx); +		if (flags != NULL) { +			iprintf(indent, "DSC content_reference_id_record_flag:%i content_time_base_indicator:%02x\n", +				flags->content_reference_id_record_flag, +				flags->content_time_base_indicator); + +			struct mpeg_content_labelling_descriptor_reference_id *reference_id = +				mpeg_content_labelling_descriptor_reference_id(flags); +			if (reference_id != NULL) { +				hexdump(indent, "DSC reference_id " , +					mpeg_content_reference_id_data(reference_id), +					reference_id->content_reference_id_record_length); +			} + +			struct mpeg_content_labelling_descriptor_time_base *time_base = +				mpeg_content_labelling_descriptor_time_base(flags); +			if (time_base != NULL) { +				iprintf(indent, "DSC time_base content_time_base_value:%lli metadata_time_base_value:%lli\n", +					time_base->content_time_base_value, +					time_base->metadata_time_base_value); +			} + +			struct mpeg_content_labelling_descriptor_content_id *content_id = +				mpeg_content_labelling_descriptor_content_id(flags); +			if (content_id != NULL) { +				iprintf(indent, "DSC content_id contentId:%i\n", +					content_id->contentId); +			} + +			struct mpeg_content_labelling_descriptor_time_base_association *time_base_assoc = +				mpeg_content_labelling_descriptor_time_base_assoc(flags); +			if (time_base_assoc != NULL) { +				hexdump(indent, "DSC time_base_assoc" , +					mpeg_time_base_association_data(time_base_assoc), +					time_base_assoc->time_base_association_data_length); +			} + +			uint8_t *priv; +			int priv_length; +			priv = mpeg_content_labelling_descriptor_data(dx, flags, &priv_length); +			hexdump(indent, "DSC private_data", priv, priv_length); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_metadata_pointer: +	{ +		struct mpeg_metadata_pointer_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_metadata_pointer_descriptor\n"); +		dx = mpeg_metadata_pointer_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_metadata_pointer_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC metadata_application_format:%04x\n", +			dx->metadata_application_format); + +		struct mpeg_metadata_pointer_descriptor_application_format_identifier *id = +			mpeg_metadata_pointer_descriptor_appid(dx); +		if (id != NULL) { +			iprintf(indent, "DSC application_format_id:%04x\n", +				id->id); +		} + +		struct mpeg_metadata_pointer_descriptor_format_identifier *did = +			mpeg_metadata_pointer_descriptor_formid(dx); +		if (did != NULL) { +			iprintf(indent, "DSC mpeg_metadata_pointer_descriptor_format_id:%04x\n", +				did->id); +		} + +		struct mpeg_metadata_pointer_descriptor_flags *flags = +			mpeg_metadata_pointer_descriptor_flags(dx); +		if (flags != NULL) { +			iprintf(indent, "DSC metadata_service_id:%i metadata_locator_record_flag:%i mpeg_carriage_flags:%x\n", +				flags->metadata_service_id, +				flags->metadata_locator_record_flag, +				flags->mpeg_carriage_flags); + +			struct mpeg_metadata_pointer_descriptor_locator *locator = +				mpeg_metadata_pointer_descriptor_locator(flags); +			if (locator != NULL) { +				hexdump(indent, "DSC locator" , +					mpeg_metadata_pointer_descriptor_locator_data(locator), +					locator->metadata_locator_record_length); +			} + +			struct mpeg_metadata_pointer_descriptor_program_number *pnum= +				mpeg_metadata_pointer_descriptor_program_number(flags); +			if (pnum != NULL) { +				iprintf(indent, "DSC program_number number:%04x\n", +					pnum->number); +			} + +			struct mpeg_metadata_pointer_descriptor_carriage *carriage = +				mpeg_metadata_pointer_descriptor_carriage(flags); +			if (carriage != NULL) { +				iprintf(indent, "DSC carriage transport_stream_location:%04x transport_stream_id:%04x\n", +					carriage->transport_stream_location, +					carriage->transport_stream_id); +			} + +			uint8_t *priv; +			int priv_length; +			priv = mpeg_metadata_pointer_descriptor_private_data(dx, flags, &priv_length); +			hexdump(indent, "DSC private_data" , priv, priv_length); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_metadata: +	{ +		struct mpeg_metadata_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_metadata_descriptor\n"); +		dx = mpeg_metadata_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_metadata_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC metadata_application_format:%04x\n", +			dx->metadata_application_format); + +		struct mpeg_metadata_descriptor_application_format_identifier *id = +			mpeg_metadata_descriptor_appid(dx); +		if (id != NULL) { +			iprintf(indent, "DSC application_format_id:%04x\n", +				id->id); +		} + +		struct mpeg_metadata_descriptor_format_identifier *did = +			mpeg_metadata_descriptor_formid(dx); +		if (did != NULL) { +			iprintf(indent, "DSC mpeg_metadata_descriptor_format_id:%04x\n", +				did->id); +		} + +		struct mpeg_metadata_descriptor_flags *flags = +			mpeg_metadata_descriptor_flags(dx); +		if (flags != NULL) { +			iprintf(indent, "DSC metadata_service_id:%i decoder_config_flags:%i dsm_cc_flag:%x\n", +				flags->metadata_service_id, +				flags->decoder_config_flags, +				flags->dsm_cc_flag); + +			struct mpeg_metadata_descriptor_service_identifier *serviceid= +				mpeg_metadata_descriptor_sevice_identifier(flags); +			if (serviceid != NULL) { +				hexdump(indent, "DSC service_id" , +					mpeg_metadata_descriptor_service_identifier_data(serviceid), +					serviceid->service_identification_length); +			} + +			struct mpeg_metadata_descriptor_decoder_config *dconfig= +				mpeg_metadata_descriptor_decoder_config(flags); +			if (dconfig != NULL) { +				hexdump(indent, "DSC decoder_config" , +					mpeg_metadata_descriptor_decoder_config_data(dconfig), +					dconfig->decoder_config_length); +			} + +			struct mpeg_metadata_descriptor_decoder_config_id_record *dconfigid= +				mpeg_metadata_descriptor_decoder_config_id_record(flags); +			if (dconfigid != NULL) { +				hexdump(indent, "DSC decoder_config" , +					mpeg_metadata_descriptor_decoder_config_id_record_data(dconfigid), +					dconfigid->decoder_config_id_record_length); +			} + +			struct mpeg_metadata_descriptor_decoder_config_service_id *dserviceid= +				mpeg_metadata_descriptor_decoder_config_service_id(flags); +			if (dserviceid != NULL) { +				iprintf(indent, "DSC decoder config service_id:%04x\n", +					dserviceid->decoder_config_metadata_service_id); +			} + +			struct mpeg_metadata_descriptor_decoder_config_reserved *reserved= +				mpeg_metadata_descriptor_decoder_config_reserved(flags); +			if (reserved != NULL) { +				hexdump(indent, "DSC reserved" , +					mpeg_metadata_descriptor_decoder_config_reserved_data(reserved), + 					reserved->reserved_data_length); +			} + +			uint8_t *priv; +			int priv_length; +			priv = mpeg_metadata_descriptor_private_data(dx, flags, &priv_length); +			hexdump(indent, "DSC private_data" , priv, priv_length); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_mpeg_metadata_std: +	{ +		struct mpeg_metadata_std_descriptor *dx; + +		iprintf(indent, "DSC Decode mpeg_metadata_std_descriptor\n"); +		dx = mpeg_metadata_std_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX mpeg_metadata_std_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC metadata_input_leak_rate:%i metadata_buffer_size:%i metadata_output_leak_rate:%i\n", +			dx->metadata_input_leak_rate, +			dx->metadata_buffer_size, +			dx->metadata_output_leak_rate); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	default: +		switch(data_type) { +		case DATA_TYPE_DVB: +			parse_dvb_descriptor(d, indent, data_type); +			return; + +		case DATA_TYPE_ATSC: +			parse_atsc_descriptor(d, indent, data_type); +			return; + +		default: +			fprintf(stderr, "DSC XXXX Unknown descriptor_tag:0x%02x\n", d->tag); +			hexdump(0, "DSC ", (uint8_t*) d, d->len+2); +			return; +		} +	} +} + +void parse_dvb_descriptor(struct descriptor *d, int indent, int data_type) +{ +	(void) data_type; + +	switch(d->tag) { +	case dtag_dvb_network_name: +	{ +		struct dvb_network_name_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_network_name_descriptor\n"); +		dx = dvb_network_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_network_name_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC name:%.*s\n", +			dvb_network_name_descriptor_name_length(dx), +			dvb_network_name_descriptor_name(dx)); +		break; +	} + +	case dtag_dvb_service_list: +	{ +		struct dvb_service_list_descriptor *dx; +		struct dvb_service_list_service *curs; + +		iprintf(indent, "DSC Decode dvb_service_list_descriptor\n"); +		dx = dvb_service_list_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_service_list_descriptor decode error\n"); +			return; +		} +		dvb_service_list_descriptor_services_for_each(dx, curs) { +			iprintf(indent+1, "DSC service_id:0x%04x service_type:0x%02x\n", +				curs->service_id, curs->service_type); +		} +		break; +	} + +	case dtag_dvb_stuffing: +	{ +		struct dvb_stuffing_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_stuffing_descriptor\n"); +		dx = dvb_stuffing_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_stuffing_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			dvb_stuffing_descriptor_data(dx), +			dvb_stuffing_descriptor_data_length(dx)); +		break; +	} + +	case dtag_dvb_satellite_delivery_system: +	{ +		struct dvb_satellite_delivery_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_satellite_delivery_descriptor\n"); +		dx = dvb_satellite_delivery_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_satellite_delivery_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC frequency:%i orbital_position:%i west_east:%i polarization:%i roll_off:%i modulation_system:%i modulation_type: %i symbol_rate:%i fec_inner:%i\n", +			dx->frequency, +			dx->orbital_position, +			dx->west_east_flag, +			dx->polarization, +			dx->roll_off, +			dx->modulation_system, +			dx->modulation_type, +			dx->symbol_rate, +			dx->fec_inner); +		break; +	} + +	case dtag_dvb_cable_delivery_system: +	{ +		struct dvb_cable_delivery_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_cable_delivery_descriptor\n"); +		dx = dvb_cable_delivery_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_cable_delivery_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC frequency:%i fec_outer:%i modulation:%i symbol_rate:%i fec_inner:%i\n", +			dx->frequency, dx->fec_outer, dx->modulation, +			dx->symbol_rate, dx->fec_inner); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_vbi_data: +	{ +		struct dvb_vbi_data_descriptor *dx; +		struct dvb_vbi_data_entry *cur; +		struct dvb_vbi_data_x *curx; + +		iprintf(indent, "DSC Decode dvb_vbi_data_descriptor\n"); +		dx = dvb_vbi_data_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_vbi_data_descriptor decode error\n"); +			return; +		} +		dvb_vbi_data_descriptor_entries_for_each(dx, cur) { +			curx = dvb_vbi_data_entry_data_x(cur); +			iprintf(indent+1, "DSC data_service_id:0x%04x\n", cur->data_service_id); +			if (cur == NULL) { +				hexdump(indent+1, "DSC", dvb_vbi_data_entry_data(cur), cur->data_length); +			} else { +				iprintf(indent+1, "DSC field_parity:%i line_offset:%i\n", +					curx->field_parity, curx->line_offset); +			} +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_vbi_teletext: +	{ +		struct dvb_vbi_teletext_descriptor *dx; +		struct dvb_vbi_teletext_entry *cur; + +		iprintf(indent, "DSC Decode dvb_vbi_teletext_descriptor\n"); +		dx = dvb_vbi_teletext_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_vbi_teletext_descriptor decode error\n"); +			return; +		} +		dvb_vbi_teletext_descriptor_entries_for_each(dx, cur) { +			iprintf(indent+1, "DSC language_code:%.3s type:%i magazine_number:%i page_number:%i\n", +				cur->language_code, +				cur->type, cur->magazine_number, cur->page_number); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_bouquet_name: +	{ +		struct dvb_bouquet_name_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_bouquet_name_descriptor\n"); +		dx = dvb_bouquet_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_bouquet_name_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC name:%.*s\n", +			dvb_bouquet_name_descriptor_name_length(dx), +			dvb_bouquet_name_descriptor_name(dx)); +		break; +	} + +	case dtag_dvb_service: +	{ +		struct dvb_service_descriptor *dx; +		struct dvb_service_descriptor_part2 *part2; + +		iprintf(indent, "DSC Decode dvb_service_descriptor\n"); +		dx = dvb_service_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_service_descriptor decode error\n"); +			return; +		} +		part2 = dvb_service_descriptor_part2(dx); +		iprintf(indent, "DSC service_type:%02x provider_name:%.*s service_name:%.*s\n", +			dx->service_type, +			dx->service_provider_name_length, +			dvb_service_descriptor_service_provider_name(dx), +			part2->service_name_length, +			dvb_service_descriptor_service_name(part2)); +		break; +	} + +	case dtag_dvb_country_availability: +	{ +		struct dvb_country_availability_descriptor *dx; +		struct dvb_country_availability_entry *cur; + +		iprintf(indent, "DSC Decode dvb_country_availability_descriptor\n"); +		dx = dvb_country_availability_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_country_availability_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC country_availability_flag:%i\n", dx->country_availability_flag); +		dvb_country_availability_descriptor_countries_for_each(dx, cur) { +			iprintf(indent+1, "DSC country_code:%.3s\n", cur->country_code); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_linkage: +	{ +		struct dvb_linkage_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_linkage_descriptor\n"); +		dx = dvb_linkage_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_linkage_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC transport_stream_id:0x%04x original_network_id:0x%04x service_id:0x%04x linkage_type:0x%02x\n", +			dx->transport_stream_id, dx->original_network_id, dx->service_id, dx->linkage_type); +		switch(dx->linkage_type) { +		case 0x08: +		{ +			struct dvb_linkage_data_08 *d08 = dvb_linkage_data_08(dx); +			int network_id = dvb_linkage_data_08_network_id(dx, d08); +			int initial_service_id = dvb_linkage_data_08_initial_service_id(dx, d08); +			int length = 0; +			uint8_t *data; + +			data = dvb_linkage_data_08_data(dx, d08, &length); +			iprintf(indent, "DSC hand_over_type:%i origin_type:%i\n", +				d08->hand_over_type, d08->origin_type); +			if (network_id != -1) { +				iprintf(indent, "DSC network_id:0x%04x\n", network_id); +			} +			if (initial_service_id != -1) { +				iprintf(indent, "DSC initial_service_id:0x%04x\n", initial_service_id); +			} +		} + +		case 0x0b: +		{ +			struct dvb_linkage_data_0b *data = dvb_linkage_data_0b(dx); +			struct dvb_platform_id *platid; +			struct dvb_platform_name *curplatname; + +			dvb_linkage_data_0b_platform_id_for_each(data, platid) { +				iprintf(indent+1, "DSC platform_id:0x%06x\n", platid->platform_id); +				dvb_platform_id_platform_name_for_each(platid, curplatname) { +					iprintf(indent+2, "DSC language_code:%.3s platform_name:%.*s\n", +						curplatname->language_code, +						curplatname->platform_name_length, dvb_platform_name_text(curplatname)); +				} +			} +			break; +		} + +		case 0x0c: +		{ +			struct dvb_linkage_data_0c *data = dvb_linkage_data_0c(dx); + +			iprintf(indent, "DSC table_type:0x%02x\n", data->table_type); +			if (dvb_linkage_data_0c_bouquet_id(data)) { +				iprintf(indent, "DSC bouquet_id:0x%04x\n", +					dvb_linkage_data_0c_bouquet_id(data)); +			} +			break; +		} + +		default: +			hexdump(indent+1, "DSC", dvb_linkage_descriptor_data(dx), dvb_linkage_descriptor_data_length(dx)); +			break; +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_nvod_reference: +	{ +		struct dvb_nvod_reference_descriptor *dx; +		struct dvb_nvod_reference *cur; + +		iprintf(indent, "DSC Decode dvb_nvod_reference_descriptor\n"); +		dx = dvb_nvod_reference_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_nvod_reference_descriptor decode error\n"); +			return; +		} +		dvb_nvod_reference_descriptor_references_for_each(dx, cur) { +			iprintf(indent+1, "DSC transport_stream_id:0x%04x original_network_id:0x%04x service_id:0x%04x\n", +				cur->transport_stream_id, cur->original_network_id, +				cur->service_id); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_time_shifted_service: +	{ +		struct dvb_time_shifted_service_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_time_shifted_service_descriptor\n"); +		dx = dvb_time_shifted_service_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_time_shifted_service_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC reference_service_id:0x%04x\n", dx->reference_service_id); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_short_event: +	{ +		struct dvb_short_event_descriptor *dx; +		struct dvb_short_event_descriptor_part2 *part2; + +		iprintf(indent, "DSC Decode dvb_short_event_descriptor\n"); +		dx = dvb_short_event_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_short_event_descriptor decode error\n"); +			return; +		} +		part2 = dvb_short_event_descriptor_part2(dx); +		iprintf(indent, "DSC language_code:%.3s event_name:%.*s text:%.*s\n", +			dx->language_code, +			dx->event_name_length, dvb_short_event_descriptor_event_name(dx), +			part2->text_length, dvb_short_event_descriptor_text(part2)); +		break; +	} + +	case dtag_dvb_extended_event: +	{ +		struct dvb_extended_event_descriptor *dx; +		struct dvb_extended_event_descriptor_part2 *part2; +		struct dvb_extended_event_item *cur; + +		iprintf(indent, "DSC Decode dvb_extended_event_descriptor\n"); +		dx = dvb_extended_event_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_extended_event_descriptor decode error\n"); +			return; +		} +		part2 = dvb_extended_event_descriptor_part2(dx); +		iprintf(indent, "DSC descriptor_number:%i last_descriptor_number:%i language_code:%.3s text:%.*s\n", +			dx->descriptor_number, dx->last_descriptor_number, +			dx->language_code, +			part2->text_length, dvb_extended_event_descriptor_part2_text(part2)); +		dvb_extended_event_descriptor_items_for_each(dx, cur) { +			struct dvb_extended_event_item_part2 *ipart2 = +				dvb_extended_event_item_part2(cur); +			iprintf(indent+1, "DSC description:%.*s item:%.*s\n", +				cur->item_description_length, dvb_extended_event_item_description(cur), +				ipart2->item_length, dvb_extended_event_item_part2_item(ipart2)); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_time_shifted_event: +	{ +		struct dvb_time_shifted_event_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_time_shifted_event_descriptor\n"); +		dx = dvb_time_shifted_event_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_time_shifted_event_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC reference_service_id:0x%04x reference_event_id:0x%04x\n", +			dx->reference_service_id, dx->reference_event_id); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_component: +	{ +		struct dvb_component_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_component_descriptor\n"); +		dx = dvb_component_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_component_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC stream_content:%i component_type:%i component_tag: %i language_code:%.3s, text:%.*s\n", +			dx->stream_content, +			dx->component_type, +			dx->component_tag, +			dx->language_code, +			dvb_component_descriptor_text_length(dx), +			dvb_component_descriptor_text(dx)); +		break; +	} + +	case dtag_dvb_mosaic: +	{ +		struct dvb_mosaic_descriptor *dx; +		struct dvb_mosaic_info *curinfo; + +		iprintf(indent, "DSC Decode dvb_mosaic_descriptor\n"); +		dx = dvb_mosaic_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_mosaic_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC mosaic_entry_point:%i number_of_horiz_elementary_cells:%i number_of_vert_elementary_cells:%i\n", +			dx->mosaic_entry_point, dx->number_of_horiz_elementary_cells, +			dx->number_of_vert_elementary_cells); +		dvb_mosaic_descriptor_infos_for_each(dx, curinfo) { +			struct dvb_mosaic_info_part2 *part2; +			struct dvb_mosaic_linkage *linkage; +			struct dvb_mosaic_elementary_cell_field *curfield; + +			part2 = dvb_mosaic_info_part2(curinfo); +			linkage = dvb_mosaic_linkage(part2); +			iprintf(indent+1, "DSC logical_cell_id:%i logical_cell_presentation_info:%i cell_linkage_info:0x%02x\n", +				curinfo->logical_cell_id, curinfo->logical_cell_presentation_info, +			        part2->cell_linkage_info); +			if (linkage) { +				switch(part2->cell_linkage_info) { +				case 0x01: +					iprintf(indent+1, "DSC bouquet_id:0x%04x\n", +							linkage->u.linkage_01.bouquet_id); +					break; + +				case 0x02: +					iprintf(indent+1, "DSC original_network_id:0x%04x transport_stream_id:0x%04x service_id:0x%04x\n", +							linkage->u.linkage_02.original_network_id, +							linkage->u.linkage_02.transport_stream_id, +							linkage->u.linkage_02.service_id); +					break; + +				case 0x03: +					iprintf(indent+1, "DSC original_network_id:0x%04x transport_stream_id:0x%04x service_id:0x%04x\n", +							linkage->u.linkage_03.original_network_id, +							linkage->u.linkage_03.transport_stream_id, +							linkage->u.linkage_03.service_id); +					break; + +				case 0x04: +					iprintf(indent+1, "DSC original_network_id:0x%04x transport_stream_id:0x%04x service_id:0x%04x event_id:0x%04x\n", +							linkage->u.linkage_04.original_network_id, +							linkage->u.linkage_04.transport_stream_id, +							linkage->u.linkage_04.service_id, +							linkage->u.linkage_04.event_id); +					break; +				} +			} + +			dvb_mosaic_info_fields_for_each(curinfo, curfield) { +				iprintf(indent+2, "DSC elementary_cell_id:0x%02x\n", +					curfield->elementary_cell_id); +			} +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_stream_identifier: +	{ +		struct dvb_stream_identifier_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_stream_identifier_descriptor\n"); +		dx = dvb_stream_identifier_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_stream_identifier_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC component_tag:%i\n", +			dx->component_tag); +		break; +	} + +	case dtag_dvb_ca_identifier: +	{ +		struct dvb_ca_identifier_descriptor *dx; +		int i; +		uint16_t *ids; + +		iprintf(indent, "DSC Decode dvb_ca_identifier_descriptor\n"); +		dx = dvb_ca_identifier_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_ca_identifier_descriptor decode error\n"); +			return; +		} +		ids = dvb_ca_identifier_descriptor_ca_system_ids(dx); +		for(i=0; i< dvb_ca_identifier_descriptor_ca_system_ids_count(dx); i++) { +			iprintf(indent+i, "DSC system_id:0x%04x\n", ids[i]); +		} +		break; +	} + +	case dtag_dvb_content: +	{ +		struct dvb_content_descriptor *dx; +		struct dvb_content_nibble *cur; + +		iprintf(indent, "DSC Decode dvb_content_descriptor\n"); +		dx = dvb_content_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_content_descriptor decode error\n"); +			return; +		} +		dvb_content_descriptor_nibbles_for_each(dx, cur) { +			iprintf(indent+1, "DSC content_nibble_level_1:%i content_nibble_level_2:%i user_nibble_1:%i user_nibble_2:%i\n", +				cur->content_nibble_level_1, cur->content_nibble_level_2, +				cur->user_nibble_1, cur->user_nibble_2); +		} +		break; +	} + +	case dtag_dvb_parental_rating: +	{ +		struct dvb_parental_rating_descriptor *dx; +		struct dvb_parental_rating *cur; + +		iprintf(indent, "DSC Decode dvb_parental_rating_descriptor\n"); +		dx = dvb_parental_rating_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_parental_rating_descriptor decode error\n"); +			return; +		} +		dvb_parental_rating_descriptor_ratings_for_each(dx, cur) { +			iprintf(indent+1, "DSC country_code:%.3s rating:%i\n", +				cur->country_code, cur->rating); +		} +		break; +	} + +	case dtag_dvb_teletext: +	{ +		struct dvb_teletext_descriptor *dx; +		struct dvb_teletext_entry *cur; + +		iprintf(indent, "DSC Decode dvb_teletext_descriptor\n"); +		dx = dvb_teletext_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_teletext_descriptor decode error\n"); +			return; +		} +		dvb_teletext_descriptor_entries_for_each(dx, cur) { +			iprintf(indent+1, "DSC language_code:%.3s type:%i magazine_number:%i page_number:%i\n", +				cur->language_code, +				cur->type, cur->magazine_number, cur->page_number); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_telephone: +	{ +		struct dvb_telephone_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_telephone_descriptor\n"); +		dx = dvb_telephone_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_telephone_descriptor decode error\n"); +			return; +		} +		iprintf(indent, +			"DSC foreign_availability:%i connection_type:%i country_prefix:%.*s " +			"international_area_code:%.*s operator_code:%.*s national_area_code:%.*s core_number:%.*s\n", +			dx->foreign_availability, dx->connection_type, +			dx->country_prefix_length, dvb_telephone_descriptor_country_prefix(dx), +			dx->international_area_code_length, dvb_telephone_descriptor_international_area_code(dx), +			dx->operator_code_length, dvb_telephone_descriptor_operator_code(dx), +			dx->national_area_code_length, dvb_telephone_descriptor_national_area_code(dx), +			dx->core_number_length, dvb_telephone_descriptor_core_number(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_local_time_offset: +	{ +		struct dvb_local_time_offset_descriptor *dx; +		struct dvb_local_time_offset *cur; + +		iprintf(indent, "DSC Decode dvb_local_time_offset_descriptor\n"); +		dx = dvb_local_time_offset_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_local_time_offset_descriptor decode error\n"); +			return; +		} +		dvb_local_time_offset_descriptor_offsets_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC country_code:%.3s country_region_id:%i " +				"local_time_offset_polarity:%i local_time_offset:%i " +				"time_of_change:%i next_time_offset:%i\n", +				cur->country_code, cur->country_region_id, +				cur->local_time_offset_polarity, +				dvbhhmm_to_seconds(cur->local_time_offset), +				dvbdate_to_unixtime(cur->time_of_change), +				dvbhhmm_to_seconds(cur->next_time_offset)); +		} +		break; +	} + +	case dtag_dvb_subtitling: +	{ +		struct dvb_subtitling_descriptor *dx; +		struct dvb_subtitling_entry *cur; + +		iprintf(indent, "DSC Decode dvb_subtitling_descriptor\n"); +		dx = dvb_subtitling_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_subtitling_descriptor decode error\n"); +			return; +		} +		dvb_subtitling_descriptor_subtitles_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC language_code:%.3s subtitling_type:0x%02x composition_page_id:0x%04x ancillary_page_id:0x%04x\n", +				cur->language_code, cur->subtitling_type, +				cur->composition_page_id, cur->ancillary_page_id); +		} +		break; +	} + +	case dtag_dvb_terrestial_delivery_system: +	{ +		struct dvb_terrestrial_delivery_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_terrestrial_delivery_descriptor\n"); +		dx = dvb_terrestrial_delivery_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_terrestrial_delivery_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC centre_frequency:%i bandwidth:%i priority:%i " +				"time_slicing_indicator:%i mpe_fec_indicator:%i constellation:%i " +				"hierarchy_information:%i code_rate_hp_stream:%i " +				"code_rate_lp_stream:%i guard_interval:%i transmission_mode:%i " +				"other_frequency_flag:%i\n", +			dx->centre_frequency, dx->bandwidth, dx->priority, +			dx->time_slicing_indicator, dx->mpe_fec_indicator, +			dx->constellation, +			dx->hierarchy_information, dx->code_rate_hp_stream, +			dx->code_rate_lp_stream, dx->guard_interval, +			dx->transmission_mode, dx->other_frequency_flag); +		break; +	} + +	case dtag_dvb_multilingual_network_name: +	{ +		struct dvb_multilingual_network_name_descriptor *dx; +		struct dvb_multilingual_network_name *cur; + +		iprintf(indent, "DSC Decode dvb_multilingual_network_name_descriptor\n"); +		dx = dvb_multilingual_network_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_multilingual_network_name_descriptor decode error\n"); +			return; +		} +		dvb_multilingual_network_name_descriptor_names_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC language_code:%.3s network_name:%.*s\n", +				cur->language_code, +				cur->network_name_length, +				dvb_multilingual_network_name_name(cur)); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_multilingual_bouquet_name: +	{ +		struct dvb_multilingual_bouquet_name_descriptor *dx; +		struct dvb_multilingual_bouquet_name *cur; + +		iprintf(indent, "DSC Decode dvb_multilingual_bouquet_name_descriptor\n"); +		dx = dvb_multilingual_bouquet_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_multilingual_bouquet_name_descriptor decode error\n"); +			return; +		} +		dvb_multilingual_bouquet_name_descriptor_names_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC language_code:%.3s bouquet_name:%.*s\n", +				cur->language_code, +				cur->bouquet_name_length, +				dvb_multilingual_bouquet_name_name(cur)); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_multilingual_service_name: +	{ +		struct dvb_multilingual_service_name_descriptor *dx; +		struct dvb_multilingual_service_name *cur; + +		iprintf(indent, "DSC Decode dvb_multilingual_service_name_descriptor\n"); +		dx = dvb_multilingual_service_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_multilingual_service_name_descriptor decode error\n"); +			return; +		} +		dvb_multilingual_service_name_descriptor_names_for_each(dx, cur) { +			struct dvb_multilingual_service_name_part2 *part2; +			part2 = dvb_multilingual_service_name_part2(cur); + +			iprintf(indent+1, +				"DSC language_code:%.3s provider_name:%.*s service_name:%.*s\n", +				cur->language_code, +				cur->service_provider_name_length, +				dvb_multilingual_service_name_service_provider_name(cur), +				part2->service_name_length, +				dvb_multilingual_service_name_service_name(part2)); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_multilingual_component: +	{ +		struct dvb_multilingual_component_descriptor *dx; +		struct dvb_multilingual_component *cur; + +		iprintf(indent, "DSC Decode dvb_multilingual_component_descriptor\n"); +		dx = dvb_multilingual_component_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_multilingual_component_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC component_tag:%02x\n", dx->component_tag); +		dvb_multilingual_component_descriptor_components_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC language_code:%.3s description:%.*s\n", +				cur->language_code, +				cur->text_description_length, +				dvb_multilingual_component_text_char(cur)); +		} +		break; +	} + +	case dtag_dvb_private_data_specifier: +	{ +		struct dvb_private_data_specifier_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_private_data_specifier_descriptor\n"); +		dx = dvb_private_data_specifier_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_private_data_specifier_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC private_data_specifier:0x%08x\n", +			dx->private_data_specifier); +		break; +	} + +	case dtag_dvb_service_move: +	{ +		struct dvb_service_move_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_service_move_descriptor\n"); +		dx = dvb_service_move_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_service_move_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC new_original_network_id:0x%04x new_transport_stream_id:0x%04x new_service_id:0x%04x\n", +			dx->new_original_network_id, dx->new_transport_stream_id, dx->new_service_id); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_short_smoothing_buffer: +	{ +		struct dvb_short_smoothing_buffer_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_short_smoothing_buffer_descriptor\n"); +		dx = dvb_short_smoothing_buffer_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_short_smoothing_buffer_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC sb_size:%i sb_leak_rate:%i\n", +			dx->sb_size, dx->sb_leak_rate); +		hexdump(indent, "DSC", +			dvb_short_smoothing_buffer_descriptor_reserved(dx), +			dvb_short_smoothing_buffer_descriptor_reserved_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_frequency_list: +	{ +		struct dvb_frequency_list_descriptor *dx; +		uint32_t *freqs; +		int count; +		int i; + +		iprintf(indent, "DSC Decode dvb_frequency_list_descriptor\n"); +		dx = dvb_frequency_list_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_frequency_list_descriptor decode error\n"); +			return; +		} +		iprintf(0, "DSC coding_type=%i\n", dx->coding_type); + +		freqs = dvb_frequency_list_descriptor_centre_frequencies(dx); +		count = dvb_frequency_list_descriptor_centre_frequencies_count(dx); +		for(i=0; i< count; i++) { +			iprintf(indent+1, "DSC %i\n", freqs[i]); +		} +		break; +	} + +	case dtag_dvb_partial_transport_stream: +	{ +		struct dvb_partial_transport_stream_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_partial_transport_stream_descriptor\n"); +		dx = dvb_partial_transport_stream_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_partial_transport_stream_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC peak_rate:%i minimum_overall_smoothing_rate:%i maximum_overall_smoothing_rate:%i\n", +			dx->peak_rate, dx->minimum_overall_smoothing_rate, dx->maximum_overall_smoothing_rate); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_data_broadcast: +	{ +		struct dvb_data_broadcast_descriptor *dx; +		struct dvb_data_broadcast_descriptor_part2 *part2; + +		iprintf(indent, "DSC Decode dvb_data_broadcast_descriptor\n"); +		dx = dvb_data_broadcast_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_data_broadcast_descriptor decode error\n"); +			return; +		} +		part2 = dvb_data_broadcast_descriptor_part2(dx); + +		iprintf(indent, "DSC data_broadcast_id:0x%04x component_tag:0x%02x selector:%.*s language_code:%.3s text:%.*s\n", +			dx->data_broadcast_id, dx->component_tag, +			dx->selector_length, dvb_data_broadcast_descriptor_selector(dx), +			part2->language_code, +			part2->text_length, dvb_data_broadcast_descriptor_part2_text(part2)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_scrambling: +	{ +		struct dvb_scrambling_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_scrambling_descriptor\n"); +		dx = dvb_scrambling_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_scrambling_descriptor decode error\n"); +			return; +		} + +		iprintf(indent, "DSC scrambling_mode:0x%02x\n", +			dx->scrambling_mode); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_data_broadcast_id: +	{ +		struct dvb_data_broadcast_id_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_data_broadcast_id_descriptor\n"); +		dx = dvb_data_broadcast_id_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_data_broadcast_id_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC data_broadcast_id:0x%04x\n", +			dx->data_broadcast_id); +		hexdump(indent+1, "DSC", +			dvb_data_broadcast_id_descriptor_id_selector_byte(dx), +			dvb_data_broadcast_id_descriptor_id_selector_byte_length(dx)); +		break; +	} + +	case dtag_dvb_transport_stream: +	{ +		struct dvb_transport_stream_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_transport_stream_descriptor\n"); +		dx = dvb_transport_stream_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_transport_stream_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			dvb_transport_stream_descriptor_data(dx), +			dvb_transport_stream_descriptor_data_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_dsng: +	{ +		struct dvb_dsng_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_dsng_descriptor\n"); +		dx = dvb_dsng_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_dsng_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			dvb_dsng_descriptor_data(dx), +			dvb_dsng_descriptor_data_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_pdc: +	{ +		struct dvb_pdc_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_pdc_descriptor\n"); +		dx = dvb_pdc_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_pdc_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC programme_id_label:0x%06x\n", +			dx->programme_id_label); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_ac3: +	{ +		struct dvb_ac3_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_ac3_descriptor\n"); +		dx = dvb_ac3_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_ac3_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC ac3_type_flag:%i bsid_flag:%i mainid_flag:%i asvc_flag:%i\n", +			dx->ac3_type_flag, dx->bsid_flag, dx->mainid_flag, dx->asvc_flag); +		hexdump(indent+1, "DSC", +			dvb_ac3_descriptor_additional_info(dx), +			dvb_ac3_descriptor_additional_info_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_ancillary_data: +	{ +		struct dvb_ancillary_data_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_ancillary_data_descriptor\n"); +		dx = dvb_ancillary_data_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_ancillary_data_descriptor decode error\n"); +			return; +		} +		iprintf(indent, "DSC scale_factor_error_check:%i dab_ancillary_data:%i announcement_switching_data:%i extended_ancillary_data:%i dvd_video_ancillary_data:%i\n", +			dx->scale_factor_error_check, +			dx->dab_ancillary_data, +			dx->announcement_switching_data, +			dx->extended_ancillary_data, +			dx->dvd_video_ancillary_data); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_cell_list: +	{ +		struct dvb_cell_list_descriptor *dx; +		struct dvb_cell_list_entry *cur; +		struct dvb_subcell_list_entry *cur_subcell; + +		iprintf(indent, "DSC Decode dvb_cell_list_descriptor\n"); +		dx = dvb_cell_list_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_cell_list_descriptor decode error\n"); +			return; +		} +		dvb_cell_list_descriptor_cells_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC cell_id:%04x cell_latitude:%i cell_longitude:%i cell_extend_of_latitude:%i cell_extend_of_longitude:%i\n", +				cur->cell_id, +				cur->cell_latitude, +				cur->cell_longitude, +				cur->cell_extend_of_latitude, +				cur->cell_extend_of_longitude); + +			dvb_cell_list_entry_subcells_for_each(cur, cur_subcell) { +				iprintf(indent+2, +					"DSC cell_id_extension:%04x subcell_latitude:%i subcell_longitude:%i subcell_extend_of_latitude:%i subcell_extend_of_longitude:%i\n", +					cur_subcell->cell_id_extension, +					cur_subcell->subcell_latitude, +					cur_subcell->subcell_longitude, +					cur_subcell->subcell_extend_of_latitude, +					cur_subcell->subcell_extend_of_longitude); +			} +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_cell_frequency_link: +	{ +		struct dvb_cell_frequency_link_descriptor *dx; +		struct dvb_cell_frequency_link_cell *cur; +		struct dvb_cell_frequency_link_cell_subcell *cur_subcell; + +		iprintf(indent, "DSC Decode dvb_cell_frequency_link_descriptor\n"); +		dx = dvb_cell_frequency_link_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_cell_frequency_link_descriptor decode error\n"); +			return; +		} +		dvb_cell_frequency_link_descriptor_cells_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC cell_id:%04x frequency:%i\n", +				cur->cell_id, +				cur->frequency); + +			dvb_cell_frequency_link_cell_subcells_for_each(cur, cur_subcell) { +				iprintf(indent+2, +					"DSC cell_id_extension:%04x transposer_frequency:%i\n", +					cur_subcell->cell_id_extension, +					cur_subcell->transposer_frequency); +			} +		} +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_announcement_support: +	{ +		struct dvb_announcement_support_descriptor *dx; +		struct dvb_announcement_support_entry *cur; +		struct dvb_announcement_support_reference *ref; + +		iprintf(indent, "DSC Decode dvb_announcement_support_descriptor\n"); +		dx = dvb_announcement_support_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_announcement_support_descriptor decode error\n"); +			return; +		} +		iprintf(indent, +			"DSC announcement_support_indicator:%04x\n", +			dx->announcement_support_indicator); + +		dvb_announcement_support_descriptor_entries_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC announcement_type:%i reference_type:%i\n", +				cur->announcement_type, +				cur->reference_type); + +			ref = dvb_announcement_support_entry_reference(cur); +			if (ref) { +				iprintf(indent+1, +					"DSC original_network_id:%04x transport_stream_id:%04x service_id:%04x component_tag:%02x\n", +					ref->original_network_id, +					ref->transport_stream_id, +					ref->service_id, +					ref->component_tag); +			} +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_application_signalling: +	{ +		struct dvb_application_signalling_descriptor *dx; +		struct dvb_application_signalling_entry *cur; + +		iprintf(indent, "DSC Decode dvb_application_signalling_descriptor\n"); +		dx = dvb_application_signalling_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_application_signalling_descriptor decode error\n"); +			return; +		} + +		dvb_application_signalling_descriptor_entries_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC application_type:%i AIT_version_number:%i\n", +				cur->application_type, +				cur->AIT_version_number); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_adaptation_field_data: +	{ +		struct dvb_adaptation_field_data_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_adaptation_field_data_descriptor\n"); +		dx = dvb_adaptation_field_data_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_adaptation_field_data_descriptor decode error\n"); +			return; +		} +		iprintf(indent, +			"DSC announcement_switching_data:%i\n", +			dx->announcement_switching_data); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_service_identifier: +	{ +		struct dvb_service_identifier_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_service_identifier_descriptor\n"); +		dx = dvb_service_identifier_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_service_identifier_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			dvb_service_identifier_descriptor_identifier(dx), +			dvb_service_identifier_descriptor_identifier_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_service_availability: +	{ +		struct dvb_service_availability_descriptor *dx; +		uint16_t *cellids; +		int count; +		int i; + +		iprintf(indent, "DSC Decode dvb_service_availability_descriptor\n"); +		dx = dvb_service_availability_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_service_availability_descriptor decode error\n"); +			return; +		} +		iprintf(indent, +			"DSC availability_flag:%i\n", +			dx->availability_flag); + +		cellids = dvb_service_availability_descriptor_cell_ids(dx); +		count = dvb_service_availability_descriptor_cell_ids_count(dx); +		for(i=0; i< count; i++) { +			iprintf(indent+1, "DSC", "%04x\n", cellids[i]); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_default_authority: +	{ +		struct dvb_default_authority_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_default_authority_descriptor\n"); +		dx = dvb_default_authority_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_default_authority_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			dvb_default_authority_descriptor_name(dx), +			dvb_default_authority_descriptor_name_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_related_content: +	{ +		struct dvb_related_content_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_related_content_descriptor\n"); +		dx = dvb_related_content_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_related_content_descriptor decode error\n"); +			return; +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_tva_id: +	{ +		struct dvb_tva_id_descriptor *dx; +		struct dvb_tva_id_entry *cur; + +		iprintf(indent, "DSC Decode dvb_tva_id_descriptor\n"); +		dx = dvb_tva_id_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_tva_id_descriptor decode error\n"); +			return; +		} + +		dvb_tva_id_descriptor_entries_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC tva_id:%04x running_status:%i\n", +				cur->tva_id, +				cur->running_status); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_content_identifier: +	{ +		struct dvb_content_identifier_descriptor *dx; +		struct dvb_content_identifier_entry *cur; +		struct dvb_content_identifier_entry_data_0 *data0; +		struct dvb_content_identifier_entry_data_1 *data1; + +		iprintf(indent, "DSC Decode dvb_tva_id_descriptor\n"); +		dx = dvb_content_identifier_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_content_identifier_descriptor decode error\n"); +			return; +		} + +		dvb_content_identifier_descriptor_entries_for_each(dx, cur) { +			iprintf(indent+1, +				"DSC crid_type:%i crid_location:%i\n", +				cur->crid_type, +				cur->crid_location); + +			data0 = dvb_content_identifier_entry_data_0(cur); +			if (data0) { +				hexdump(indent, "DSC data0", +					dvb_content_identifier_entry_data_0_data(data0), + 					data0->crid_length); +			} + +			data1 = dvb_content_identifier_entry_data_1(cur); +			if (data1) { +				iprintf(indent+1, +					"DSC crid_ref:%04x\n", +					data1->crid_ref); +			} +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_dvb_s2_satellite_delivery_descriptor: +	{ +		struct dvb_s2_satellite_delivery_descriptor *dx; + +		iprintf(indent, "DSC Decode dvb_s2_satellite_delivery_descriptor\n"); +		dx = dvb_s2_satellite_delivery_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX dvb_s2_satellite_delivery_descriptor decode error\n"); +			return; +		} + +		iprintf(indent, +			"DSC scrambling_sequence_selector:%i multiple_input_stream:%i backwards_compatability:%i\n", +			dx->scrambling_sequence_selector, +			dx->multiple_input_stream, +			dx->backwards_compatability); +		if (dx->scrambling_sequence_selector) { +			iprintf(indent, +				"DSC scrambling_sequence_index:%i\n", +				dvb_s2_satellite_delivery_descriptor_scrambling_sequence_index(dx)); +		} +		if (dx->multiple_input_stream) { +			iprintf(indent, +				"DSC input_stream_id:%i\n", +				dvb_s2_satellite_delivery_descriptor_input_stream_id(dx)); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	default: +		fprintf(stderr, "DSC XXXX Unknown descriptor_tag:0x%02x\n", d->tag); +		hexdump(0, "DSC ", (uint8_t*) d, d->len+2); +		return; +	} +} + +void parse_atsc_descriptor(struct descriptor *d, int indent, int data_type) +{ +	(void) data_type; + +	switch(d->tag) { +	case dtag_atsc_stuffing: +	{ +		struct atsc_stuffing_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_stuffing_descriptor\n"); +		dx = atsc_stuffing_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_stuffing_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			atsc_stuffing_descriptor_data(dx), +			atsc_stuffing_descriptor_data_length(dx)); +		break; +	} + +	case dtag_atsc_ac3_audio: +	{ +		struct atsc_ac3_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_ac3_descriptor\n"); +		dx = atsc_ac3_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_ac3_descriptor decode error\n"); +			return; +		} + +		iprintf(indent, +			"DSC sample_rate_code:%i bsid:%i bit_rate_code:%i surround_mode:%i bsmod:%i num_channels:%i full_svc:%i\n", +			dx->sample_rate_code, +			dx->bsid, +			dx->bit_rate_code, +			dx->surround_mode, +			dx->bsmod, +			dx->num_channels, +			dx->full_svc); + +		hexdump(indent+1, "DSC additional_info", +			atsc_ac3_descriptor_additional_info(dx), +			atsc_ac3_descriptor_additional_info_length(dx)); +		break; +	} + +	case dtag_atsc_caption_service: +	{ +		struct atsc_caption_service_descriptor *dx; +		struct atsc_caption_service_entry *cur; +		int idx; + +		iprintf(indent, "DSC Decode atsc_caption_service_descriptor\n"); +		dx = atsc_caption_service_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_caption_service_descriptor decode error\n"); +			return; +		} + +		atsc_caption_service_descriptor_entries_for_each(dx, cur, idx) { +			iprintf(indent+1, +				"DSC language_code:%.3s digital_cc:%i value:%i easy_reader:%i wide_aspect_ratio:%i\n", +				cur->language_code, +				cur->digital_cc, +				cur->value, +				cur->easy_reader, +				cur->wide_aspect_ratio); +		} +		break; +	} + +	case dtag_atsc_content_advisory: +	{ +		struct atsc_content_advisory_descriptor *dx; +		struct atsc_content_advisory_entry *cure; +		struct atsc_content_advisory_entry_dimension *curd; +		struct atsc_content_advisory_entry_part2 *part2; +		int eidx; +		int didx; + +		iprintf(indent, "DSC Decode atsc_content_advisory_descriptor\n"); +		dx = atsc_content_advisory_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_content_advisory_descriptor decode error\n"); +			return; +		} + +		atsc_content_advisory_descriptor_entries_for_each(dx, cure, eidx) { +			iprintf(indent+1, +				"DSC rating_region:%i\n", +				cure->rating_region); + +			atsc_content_advisory_entry_dimensions_for_each(cure, curd, didx) { +				iprintf(indent+2, +					"DSC rating_dimension_j:%i rating_value:%i\n", +					curd->rating_dimension_j, +				        curd->rating_value); +			} + +			part2 = atsc_content_advisory_entry_part2(cure); + +			atsctextdump("DSC description:", +					indent, +					atsc_content_advisory_entry_part2_description(part2), +					part2->rating_description_length); +		} + +		break; +	} + +	case dtag_atsc_extended_channel_name: +	{ +		struct atsc_extended_channel_name_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_extended_channel_name_descriptor\n"); +		dx = atsc_extended_channel_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_extended_channel_name_descriptor decode error\n"); +			return; +		} + +		atsctextdump("SCT text:", 1, +				atsc_extended_channel_name_descriptor_text(dx), +				atsc_extended_channel_name_descriptor_text_length(dx)); +		break; +	} + +	case dtag_atsc_service_location: +	{ +		struct atsc_service_location_descriptor *dx; +		struct atsc_caption_service_location_element *cur; +		int idx; + +		iprintf(indent, "DSC Decode atsc_service_location_descriptor\n"); +		dx = atsc_service_location_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_service_location_descriptor decode error\n"); +			return; +		} +		iprintf(indent+1, "DSC PCR_PID:%04x\n", dx->PCR_PID); + +		atsc_service_location_descriptor_elements_for_each(dx, cur, idx) { +			iprintf(indent+1, "DSC stream_type:%02x elementary_PID:%04x language_code:%.3s\n", +			        cur->stream_type, +			        cur->elementary_PID, +			        cur->language_code); +		} +		break; +	} + +	case dtag_atsc_time_shifted_service: +	{ +		struct atsc_time_shifted_service_descriptor *dx; +		struct atsc_time_shifted_service *cur; +		int idx; + +		iprintf(indent, "DSC Decode atsc_time_shifted_service_descriptor\n"); +		dx = atsc_time_shifted_service_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_time_shifted_service_descriptor decode error\n"); +			return; +		} + +		atsc_time_shifted_service_descriptor_services_for_each(dx, cur, idx) { +			iprintf(indent+1, "DSC time_shift:%i major_channel_number:%04x minor_channel_number:%04x\n", +				cur->time_shift, +				cur->major_channel_number, +				cur->minor_channel_number); +		} + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_atsc_component_name: +	{ +		struct atsc_component_name_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_component_name_descriptor\n"); +		dx = atsc_component_name_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_component_name_descriptor decode error\n"); +			return; +		} + +		atsctextdump("SCT name:", 1, +			     atsc_component_name_descriptor_text(dx), +			     atsc_component_name_descriptor_text_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_atsc_dcc_departing_request: +	{ +		struct atsc_dcc_departing_request_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_dcc_departing_request_descriptor\n"); +		dx = atsc_dcc_departing_request_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_dcc_departing_request_descriptor decode error\n"); +			return; +		} +		iprintf(indent+1, "DSC dcc_departing_request_type:%02x\n", +			dx->dcc_departing_request_type); + +		atsctextdump("SCT text:", 1, +				atsc_dcc_departing_request_descriptor_text(dx), +				atsc_dcc_departing_request_descriptor_text_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_atsc_dcc_arriving_request: +	{ +		struct atsc_dcc_arriving_request_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_dcc_arriving_request_descriptor\n"); +		dx = atsc_dcc_arriving_request_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_dcc_arriving_request_descriptor decode error\n"); +			return; +		} +		iprintf(indent+1, "DSC dcc_arriving_request_type:%02x\n", +			dx->dcc_arriving_request_type); + +		atsctextdump("SCT text:", 1, +			     atsc_dcc_arriving_request_descriptor_text(dx), +			     atsc_dcc_arriving_request_descriptor_text_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_atsc_redistribution_control: +	{ +		struct atsc_rc_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_rc_descriptor\n"); +		dx = atsc_rc_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_rc_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			atsc_rc_descriptor_info(dx), +			atsc_rc_descriptor_info_length(dx)); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_atsc_genre: +	{ +		struct atsc_genre_descriptor *dx; + +		iprintf(indent, "DSC Decode atsc_genre_descriptor\n"); +		dx = atsc_genre_descriptor_codec(d); +		if (dx == NULL) { +			fprintf(stderr, "DSC XXXX atsc_genre_descriptor decode error\n"); +			return; +		} +		hexdump(indent, "DSC", +			atsc_genre_descriptor_attributes(dx), +			dx->attribute_count); + +		hexdump(0, "XXX", (uint8_t*) d, d->len + 2); +		getchar(); +		break; +	} + +	case dtag_atsc_private_information: +		// FIXME: whats the format? + +	case dtag_atsc_content_identifier: +		// FIXME: whats the format? + +	default: +		fprintf(stderr, "DSC XXXX Unknown descriptor_tag:0x%02x\n", d->tag); +		hexdump(0, "DSC ", (uint8_t*) d, d->len+2); +		return; +	} +} + +void iprintf(int indent, char *fmt, ...) +{ +	va_list ap; + +	while(indent--) { +		printf("\t"); +	} + +	va_start(ap, fmt); +	vprintf(fmt, ap); +	va_end(ap); +} + +void hexdump(int indent, char *prefix, uint8_t *buf, int buflen) +{ +	int i; +	int j; +	int max; +	char line[512]; + +	for(i=0; i< buflen; i+=16) { +		max = 16; +		if ((i + max) > buflen) +				max = buflen - i; + +		memset(line, 0, sizeof(line)); +		memset(line + 4 + 48 + 1, ' ', 16); +		sprintf(line, "%02x: ", i); +		for(j=0; j<max; j++) { +			sprintf(line + 4 + (j*3), "%02x", buf[i+j]); +			if ((buf[i+j] > 31) && (buf[i+j] < 127)) +				line[4 + 48 + 1 + j] = buf[i+j]; +			else +				line[4 + 48 + 1 + j] = '.'; +		} + +		for(j=0; j< 4 + 48;  j++) { +			if (!line[j]) +				line[j] = ' '; +		} +		line[4+48] = '|'; + +		for(j=0; j < indent; j++) { +			printf("\t"); +		} +		printf("%s%s|\n", prefix, line); +	} +} + +void atsctextdump(char *header, int indent, struct atsc_text *atext, int len) +{ +	struct atsc_text_string *cur_string; +	struct atsc_text_string_segment *cur_segment; +	int str_idx; +	int seg_idx; + +	if (len == 0) +		return; + +	atsc_text_strings_for_each(atext, cur_string, str_idx) { +		iprintf(indent+1, "%s String %i language:%.3s\n", header, str_idx, cur_string->language_code); + +		atsc_text_string_segments_for_each(cur_string, cur_segment, seg_idx) { +			iprintf(indent+2, "Segment %i compression_type:%i mode:%i\n", +				seg_idx, +				cur_segment->compression_type, +			        cur_segment->mode); + +			hexdump(indent+2, "rawbytes ", +				atsc_text_string_segment_bytes(cur_segment), +				cur_segment->number_bytes); + +			if (cur_segment->compression_type < 0x3e) { +				uint8_t *decoded = NULL; +				size_t decodedlen = 0; +				size_t decodedpos = 0; + +				if (atsc_text_segment_decode(cur_segment, +				    			     &decoded, +							     &decodedlen, +							     &decodedpos) < 0) { +					iprintf(indent+2, "Decode error\n"); +				} else { +					hexdump(indent+2, "decoded  ", decoded, decodedpos); +				} +				if (decoded) +					free(decoded); +			} +		} +	} +} diff --git a/test/lnb.c b/test/lnb.c new file mode 100644 index 0000000..9052d1c --- /dev/null +++ b/test/lnb.c @@ -0,0 +1,101 @@ +#include <stdlib.h> +#include <string.h> +#include <ctype.h> +#include "lnb.h" + +static char *univ_desc[] = { +		"Europe", +		"10800 to 11800 MHz and 11600 to 12700 Mhz", +		"Dual LO, loband 9750, hiband 10600 MHz", +		(char *)NULL }; + +static char *dbs_desc[] = { +		"Expressvu, North America", +		"12200 to 12700 MHz", +		"Single LO, 11250 MHz", +		(char *)NULL }; + +static char *standard_desc[] = { +		"10945 to 11450 Mhz", +		"Single LO, 10000 Mhz", +		(char *)NULL }; + +static char *enhan_desc[] = { +		"Astra", +		"10700 to 11700 MHz", +		"Single LO, 9750 MHz", +		(char *)NULL }; + +static char *cband_desc[] = { +		"Big Dish", +		"3700 to 4200 MHz", +		"Single LO, 5150 Mhz", +		(char *)NULL }; + +static struct lnb_types_st lnbs[] = { +	{"UNIVERSAL",	univ_desc,		9750, 10600, 11700 }, + 	{"DBS",		dbs_desc, 		11250, 0, 0 }, +	{"STANDARD",	standard_desc,		10000, 0, 0 }, +	{"ENHANCED",	enhan_desc,		9750, 0, 0 }, +	{"C-BAND",	cband_desc,		5150, 0, 0 } +}; + +/* Enumerate through standard types of LNB's until NULL returned. + * Increment curno each time + */ + +struct lnb_types_st * +lnb_enum(int curno) +{ +	if (curno >= (int) (sizeof(lnbs) / sizeof(lnbs[0]))) +		return (struct lnb_types_st *)NULL; +	return &lnbs[curno]; +} + +/* Decode an lnb type, for example given on a command line + * If alpha and standard type, e.g. "Universal" then match that + * otherwise low[,high[,switch]] + */ + +int +lnb_decode(char *str, struct lnb_types_st *lnbp) +{ +int i; +char *cp, *np; + +	memset(lnbp, 0, sizeof(*lnbp)); +	cp = str; +	while(*cp && isspace(*cp)) +		cp++; +	if (isalpha(*cp)) { +		for (i = 0; i < (int)(sizeof(lnbs) / sizeof(lnbs[0])); i++) { +			if (!strcasecmp(lnbs[i].name, cp)) { +				*lnbp = lnbs[i]; +				return 1; +			} +		} +		return -1; +	} +	if (*cp == '\0' || !isdigit(*cp)) +		return -1; +	lnbp->low_val = strtoul(cp, &np, 0); +	if (lnbp->low_val == 0) +		return -1; +	cp = np; +	while(*cp && (isspace(*cp) || *cp == ',')) +		cp++; +	if (*cp == '\0') +		return 1; +	if (!isdigit(*cp)) +		return -1; +	lnbp->high_val = strtoul(cp, &np, 0); +	cp = np; +	while(*cp && (isspace(*cp) || *cp == ',')) +		cp++; +	if (*cp == '\0') +		return 1; +	if (!isdigit(*cp)) +		return -1; +	lnbp->switch_val = strtoul(cp, NULL, 0); +	return 1; +} diff --git a/test/lnb.h b/test/lnb.h new file mode 100644 index 0000000..6370fd4 --- /dev/null +++ b/test/lnb.h @@ -0,0 +1,22 @@ +struct lnb_types_st { +	char	*name; +	char	**desc; +	unsigned long	low_val; +	unsigned long	high_val;	/* zero indicates no hiband */ +	unsigned long	switch_val;	/* zero indicates no hiband */ +}; + +/* Enumerate through standard types of LNB's until NULL returned. + * Increment curno each time + */ + +struct lnb_types_st * +lnb_enum(int curno); + +/* Decode an lnb type, for example given on a command line + * If alpha and standard type, e.g. "Universal" then match that + * otherwise low[,high[,switch]] + */ + +int +lnb_decode(char *str, struct lnb_types_st *lnbp); diff --git a/test/sendburst.c b/test/sendburst.c index a96b68c..b19df37 100644 --- a/test/sendburst.c +++ b/test/sendburst.c @@ -1,8 +1,9 @@ -/* - * Test sending the burst mini command A/B on a SAT frontend. - * - * usage: FRONTEND=/dev/dvb/adapterX/frontendX sendburst {a|b} - */ +#define USAGE \ +"\n" \ +"\nTest sending the burst mini command A/B on a SAT frontend." \ +"\n" \ +"\nusage: FRONTEND=/dev/dvb/adapterX/frontendX sendburst {a|b}" \ +"\n"  #include <stdlib.h>  #include <stdio.h> @@ -22,7 +23,7 @@ int main (int argc, char **argv)     int fd, r;     if (argc != 2 || (strcmp(argv[1], "a") && strcmp(argv[1], "b"))) { -      fprintf (stderr, "usage: %s <a|b>\n", argv[0]); +      fprintf (stderr, "usage: %s <a|b>\n" USAGE, argv[0]);        return 1;     } @@ -52,4 +53,3 @@ int main (int argc, char **argv)     return 0;  } - diff --git a/test/set22k.c b/test/set22k.c index 51ffa1c..a9f0d8c 100644 --- a/test/set22k.c +++ b/test/set22k.c @@ -1,10 +1,11 @@ -/* - * Test switching the 22kHz tone signal on and off on a SAT frontend. - * (Note: DiSEqC equipment ignores this after it has once seen a diseqc - *  sequence; reload the driver or unplug/replug the SAT cable to reset.) - * - * usage: FRONTEND=/dev/dvb/adapterX/frontendX set22k {on|off} - */ +#define USAGE \ +"\n" \ +"\nTest switching the 22kHz tone signal on and off on a SAT frontend." \ +"\n(Note: DiSEqC equipment ignores this after it has once seen a diseqc" \ +"\n sequence; reload the driver or unplug/replug the SAT cable to reset.)" \ +"\n" \ +"\nusage: FRONTEND=/dev/dvb/adapterX/frontendX set22k {on|off}" \ +"\n"  #include <stdlib.h>  #include <stdio.h> @@ -24,7 +25,7 @@ int main (int argc, char **argv)     int fd, r;     if (argc != 2 || (strcmp(argv[1], "on") && strcmp(argv[1], "off"))) { -      fprintf (stderr, "usage: %s <on|off>\n", argv[0]); +      fprintf (stderr, "usage: %s <on|off>\n" USAGE, argv[0]);        return 1;     }     if (getenv("FRONTEND")) @@ -47,4 +48,3 @@ int main (int argc, char **argv)     return 0;  } - diff --git a/test/setpid.c b/test/setpid.c index fa0333c..a2d42cf 100644 --- a/test/setpid.c +++ b/test/setpid.c @@ -1,9 +1,10 @@ -/* - * Set video and audio PIDs in the demux; useful only if you have - * a hardware MPEG decoder and you're tuned to a transport stream. - * - * usage: DEMUX=/dev/dvb/adapterX/demuxX setpid video_pid audio_pid - */ +#define USAGE \ +"\n" \ +"\nSet video and audio PIDs in the demux; useful only if you have" \ +"\na hardware MPEG decoder and you're tuned to a transport stream." \ +"\n" \ +"\nusage: DEMUX=/dev/dvb/adapterX/demuxX setpid video_pid audio_pid" \ +"\n"  #include <unistd.h>  #include <stdlib.h> @@ -69,7 +70,7 @@ int main (int argc, char **argv)     int video_pid, audio_pid;     if (argc != 3) { -      printf ("\nusage: %s <video pid> <audio pid>\n\n", argv[0]); +      printf ("\nusage: %s <video pid> <audio pid>\n\n" USAGE, argv[0]);        exit (1);     }     if (getenv("DEMUX")) @@ -83,5 +84,3 @@ int main (int argc, char **argv)     return 0;  } - - diff --git a/test/setvoltage.c b/test/setvoltage.c index 1d14a4c..80f7688 100644 --- a/test/setvoltage.c +++ b/test/setvoltage.c @@ -1,10 +1,11 @@ -/* - * Test switching the voltage signal high and low on a SAT frontend. - * (Note: DiSEqC equipment ignores this after it has once seen a diseqc - *  sequence; reload the driver or unplug/replug the SAT cable to reset.) - * - * usage: FRONTEND=/dev/dvb/adapterX/frontendX setvoltage {13|18} - */ +#define USAGE \ +"\n" \ +"\nTest switching the voltage signal high and low on a SAT frontend." \ +"\n(Note: DiSEqC equipment ignores this after it has once seen a diseqc" \ +"\n sequence; reload the driver or unplug/replug the SAT cable to reset.)" \ +"\n" \ +"\nusage: FRONTEND=/dev/dvb/adapterX/frontendX setvoltage {13|18}" \ +"\n"  #include <stdlib.h>  #include <stdio.h> @@ -23,7 +24,7 @@ int main (int argc, char **argv)     int fd, r;     if (argc != 2 || (strcmp(argv[1], "13") && strcmp(argv[1], "18"))) { -      fprintf (stderr, "usage: %s <13|18>\n", argv[0]); +      fprintf (stderr, "usage: %s <13|18>\n" USAGE, argv[0]);        return -1;     }     if (getenv("FRONTEND")) @@ -44,4 +45,3 @@ int main (int argc, char **argv)     return 0;  } - diff --git a/test/szap2.c b/test/szap2.c new file mode 100644 index 0000000..e470229 --- /dev/null +++ b/test/szap2.c @@ -0,0 +1,767 @@ +/* szap -- simple zapping tool for the Linux DVB API + * + * szap operates on VDR (http://www.cadsoft.de/people/kls/vdr/index.htm) + * satellite channel lists (e.g. from http://www.dxandy.de/cgi-bin/dvbchan.pl). + * szap assumes you have a "Universal LNB" (i.e. with LOFs 9750/10600 MHz). + * + * Compilation: `gcc -Wall -I../../ost/include -O2 szap.c -o szap` + *  or, if your DVB driver is in the kernel source tree: + *              `gcc -Wall -DDVB_IN_KERNEL -O2 szap.c -o szap` + * + * Copyright (C) 2001 Johannes Stezenbach (js@convergence.de) + * for convergence integrated media + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 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 General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + */ + + +#include <stdio.h> +#include <stdlib.h> +#include <limits.h> +#include <string.h> +#include <errno.h> +#include <sys/ioctl.h> +#include <sys/types.h> +#include <sys/stat.h> +#include <sys/poll.h> +#include <sys/param.h> +#include <fcntl.h> +#include <time.h> +#include <unistd.h> + +#include <stdint.h> +#include <sys/time.h> + +#include "../include/frontend.h" +#include "../include/dmx.h" +#include "../include/audio.h" +#include "../include/version.h" +#include "lnb.h" + +#ifndef TRUE +#define TRUE (1==1) +#endif +#ifndef FALSE +#define FALSE (1==0) +#endif + +/* location of channel list file */ +#define CHANNEL_FILE "channels.conf" + +/* one line of the VDR channel file has the following format: + * ^name:frequency_MHz:polarization:sat_no:symbolrate:vpid:apid:?:service_id$ + */ + + +#define FRONTENDDEVICE "/dev/dvb/adapter%d/frontend%d" +#define DEMUXDEVICE "/dev/dvb/adapter%d/demux%d" +#define AUDIODEVICE "/dev/dvb/adapter%d/audio%d" + +static struct lnb_types_st lnb_type; + +static int exit_after_tuning; +static int interactive; + +static char *usage_str = +    "\nusage: szap -q\n" +    "         list known channels\n" +    "       szap [options] {-n channel-number|channel_name}\n" +    "         zap to channel via number or full name (case insensitive)\n" +    "     -a number : use given adapter (default 0)\n" +    "     -f number : use given frontend (default 0)\n" +    "     -d number : use given demux (default 0)\n" +    "     -c file   : read channels list from 'file'\n" +    "     -b        : enable Audio Bypass (default no)\n" +    "     -x        : exit after tuning\n" +    "     -r        : set up /dev/dvb/adapterX/dvr0 for TS recording\n" +    "     -l lnb-type (DVB-S Only) (use -l help to print types) or \n" +    "     -l low[,high[,switch]] in Mhz\n" +    "     -i        : run interactively, allowing you to type in channel names\n" +    "     -p        : add pat and pmt to TS recording (implies -r)\n" +    "                 or -n numbers for zapping\n" +    "	  -t	    : delivery system type DVB-S=0, DSS=1, DVB-S2=2\n"; + +static int set_demux(int dmxfd, int pid, int pes_type, int dvr) +{ +	struct dmx_pes_filter_params pesfilter; + +	if (pid < 0 || pid >= 0x1fff) /* ignore this pid to allow radio services */ +		return TRUE; + +	if (dvr) { +		int buffersize = 64 * 1024; +		if (ioctl(dmxfd, DMX_SET_BUFFER_SIZE, buffersize) == -1) +			perror("DMX_SET_BUFFER_SIZE failed"); +	} + +	pesfilter.pid = pid; +	pesfilter.input = DMX_IN_FRONTEND; +	pesfilter.output = dvr ? DMX_OUT_TS_TAP : DMX_OUT_DECODER; +	pesfilter.pes_type = pes_type; +	pesfilter.flags = DMX_IMMEDIATE_START; + +	if (ioctl(dmxfd, DMX_SET_PES_FILTER, &pesfilter) == -1) { +		fprintf(stderr, "DMX_SET_PES_FILTER failed " +			"(PID = 0x%04x): %d %m\n", pid, errno); + +		return FALSE; +	} + +	return TRUE; +} + +int get_pmt_pid(char *dmxdev, int sid) +{ +	int patfd, count; +	int pmt_pid = 0; +	int patread = 0; +	int section_length; +	unsigned char buft[4096]; +	unsigned char *buf = buft; +	struct dmx_sct_filter_params f; + +	memset(&f, 0, sizeof(f)); +	f.pid = 0; +	f.filter.filter[0] = 0x00; +	f.filter.mask[0] = 0xff; +	f.timeout = 0; +	f.flags = DMX_IMMEDIATE_START | DMX_CHECK_CRC; + +	if ((patfd = open(dmxdev, O_RDWR)) < 0) { +		perror("openening pat demux failed"); +		return -1; +	} + +	if (ioctl(patfd, DMX_SET_FILTER, &f) == -1) { +		perror("ioctl DMX_SET_FILTER failed"); +		close(patfd); +		return -1; +	} + +	while (!patread) { +		if (((count = read(patfd, buf, sizeof(buft))) < 0) && errno == EOVERFLOW) +			count = read(patfd, buf, sizeof(buft)); +		if (count < 0) { +			perror("read_sections: read error"); +			close(patfd); +			return -1; +		} + +		section_length = ((buf[1] & 0x0f) << 8) | buf[2]; +		if (count != section_length + 3) +			continue; + +		buf += 8; +		section_length -= 8; + +		patread = 1; /* assumes one section contains the whole pat */ +		while (section_length > 0) { +			int service_id = (buf[0] << 8) | buf[1]; +			if (service_id == sid) { +				pmt_pid = ((buf[2] & 0x1f) << 8) | buf[3]; +				section_length = 0; +			} +			buf += 4; +			section_length -= 4; +		} +	} +	close(patfd); +	return pmt_pid; +} + +struct diseqc_cmd { +	struct dvb_diseqc_master_cmd cmd; +	uint32_t wait; +}; + +void diseqc_send_msg(int fd, fe_sec_voltage_t v, struct diseqc_cmd *cmd, +		     fe_sec_tone_mode_t t, fe_sec_mini_cmd_t b) +{ +	if (ioctl(fd, FE_SET_TONE, SEC_TONE_OFF) == -1) +		perror("FE_SET_TONE failed"); +	if (ioctl(fd, FE_SET_VOLTAGE, v) == -1) +		perror("FE_SET_VOLTAGE failed"); +		usleep(15 * 1000); +	if (ioctl(fd, FE_DISEQC_SEND_MASTER_CMD, &cmd->cmd) == -1) +		perror("FE_DISEQC_SEND_MASTER_CMD failed"); +		usleep(cmd->wait * 1000); +		usleep(15 * 1000); +	if (ioctl(fd, FE_DISEQC_SEND_BURST, b) == -1) +		perror("FE_DISEQC_SEND_BURST failed"); +		usleep(15 * 1000); +	if (ioctl(fd, FE_SET_TONE, t) == -1) +		perror("FE_SET_TONE failed"); +} + + + + +/* digital satellite equipment control, + * specification is available from http://www.eutelsat.com/ + */ +static int diseqc(int secfd, int sat_no, int pol_vert, int hi_band) +{ +	struct diseqc_cmd cmd = +		{ {{0xe0, 0x10, 0x38, 0xf0, 0x00, 0x00}, 4}, 0 }; + +	/** +	 * param: high nibble: reset bits, low nibble set bits, +	 * bits are: option, position, polarizaion, band +	 */ +	cmd.cmd.msg[3] = +		0xf0 | (((sat_no * 4) & 0x0f) | (hi_band ? 1 : 0) | (pol_vert ? 0 : 2)); + +	diseqc_send_msg(secfd, pol_vert ? SEC_VOLTAGE_13 : SEC_VOLTAGE_18, +			&cmd, hi_band ? SEC_TONE_ON : SEC_TONE_OFF, +			(sat_no / 4) % 2 ? SEC_MINI_B : SEC_MINI_A); + +	return TRUE; +} + +#define DVBS	0 +#define DSS	1 +#define DVBS2	2 + +static int do_tune(int fefd, unsigned int ifreq, unsigned int sr, unsigned int delsys) +{ +	/*	API Major=3, Minor=1	*/ +	struct dvb_frontend_parameters tuneto; +	struct dvb_frontend_event ev; +	/*	API Major=3, Minor=2	*/ +	struct dvbfe_params fe_params; + +	/* discard stale QPSK events */ +	while (1) { +		if (ioctl(fefd, FE_GET_EVENT, &ev) == -1) +		break; +	} + +	if ((DVB_API_VERSION == 3) && (DVB_API_VERSION_MINOR == 3)) { +		printf("\n%s: API version=%d, delivery system = %d\n", __func__, DVB_API_VERSION_MINOR, delsys); + +		fe_params.frequency = ifreq; +		fe_params.inversion = INVERSION_AUTO; +		 +		switch (delsys) { +		case DVBS: +			fe_params.delsys.dvbs.symbol_rate = sr; +			fe_params.delsys.dvbs.fec = FEC_AUTO; +			printf("%s: Frequency = %d, Srate = %d\n", +				__func__, fe_params.frequency, fe_params.delsys.dvbs.symbol_rate); +			break; +		case DSS: +			fe_params.delsys.dss.symbol_rate = sr; +			fe_params.delsys.dss.fec = FEC_AUTO; +			printf("%s: Frequency = %d, Srate = %d\n", +				__func__, fe_params.frequency, fe_params.delsys.dss.symbol_rate); +			break; +		case DVBS2: +			fe_params.delsys.dvbs2.symbol_rate = sr; +			fe_params.delsys.dvbs2.fec = FEC_AUTO; +			printf("%s: Frequency = %d, Srate = %d\n", +				__func__, fe_params.frequency, fe_params.delsys.dvbs2.symbol_rate); +			break; +		default: +			return -EINVAL; +		} +		printf("%s: Frequency = %d, Srate = %d\n\n\n", +			__func__, fe_params.frequency, fe_params.delsys.dvbs.symbol_rate); + +		if (ioctl(fefd, DVBFE_SET_PARAMS, &fe_params) == -1) { +			perror("DVBFE_SET_PARAMS failed"); +			return FALSE; +		} + +	} else if ((DVB_API_VERSION == 3) && (DVB_API_VERSION_MINOR == 1)){ +		tuneto.frequency = ifreq; +		tuneto.inversion = INVERSION_AUTO; +		tuneto.u.qpsk.symbol_rate = sr; +		tuneto.u.qpsk.fec_inner = FEC_AUTO; +		if (ioctl(fefd, FE_SET_FRONTEND, &tuneto) == -1) { +			perror("FE_SET_FRONTEND failed"); +			return FALSE; +		} +	} +	return TRUE; +} + + +static +int check_frontend (int fe_fd, int dvr) +{ +	(void)dvr; +	fe_status_t status; +	uint16_t snr, signal; +	uint32_t ber, uncorrected_blocks; +	int timeout = 0; + +	do { +		if (ioctl(fe_fd, FE_READ_STATUS, &status) == -1) +			perror("FE_READ_STATUS failed"); +		/* some frontends might not support all these ioctls, thus we +		 * avoid printing errors +		 */ +		if (ioctl(fe_fd, FE_READ_SIGNAL_STRENGTH, &signal) == -1) +			signal = -2; +		if (ioctl(fe_fd, FE_READ_SNR, &snr) == -1) +			snr = -2; +		if (ioctl(fe_fd, FE_READ_BER, &ber) == -1) +			ber = -2; +		if (ioctl(fe_fd, FE_READ_UNCORRECTED_BLOCKS, &uncorrected_blocks) == -1) +			uncorrected_blocks = -2; + +		printf ("status %02x | signal %04x | snr %04x | ber %08x | unc %08x | ", +			status, signal, snr, ber, uncorrected_blocks); + +		if (status & FE_HAS_LOCK) +			printf("FE_HAS_LOCK"); +		printf("\n"); + +		if (exit_after_tuning && ((status & FE_HAS_LOCK) || (++timeout >= 10))) +			break; + +		usleep(1000000); +	} while (1); + +	return 0; +} + + +static +int zap_to(unsigned int adapter, unsigned int frontend, unsigned int demux, +	   unsigned int sat_no, unsigned int freq, unsigned int pol, +	   unsigned int sr, unsigned int vpid, unsigned int apid, int sid, +	   int dvr, int rec_psi, int bypass, unsigned int delsys) +{ +	char fedev[128], dmxdev[128], auddev[128]; +	static int fefd, dmxfda, dmxfdv, audiofd = -1, patfd, pmtfd; +	int pmtpid; +	uint32_t ifreq; +	int hiband, result; +	enum dvbfe_delsys delivery; +	 +	switch (delsys) { +	case DVBS: +		printf("Delivery system=DVB-S\n"); +		delivery = DVBFE_DELSYS_DVBS;	 +		break; +	case DSS: +		printf("Delivery system=DSS\n"); +		delivery = DVBFE_DELSYS_DSS; +		break; +	case DVBS2: +		printf("Delivery system=DVB-S2\n"); +		delivery = DVBFE_DELSYS_DVBS2; +		break; +	default: +		printf("Unsupported delivery system\n"); +		return -EINVAL; +	} + +	if (!fefd) { +		snprintf(fedev, sizeof(fedev), FRONTENDDEVICE, adapter, frontend); +		snprintf(dmxdev, sizeof(dmxdev), DEMUXDEVICE, adapter, demux); +		snprintf(auddev, sizeof(auddev), AUDIODEVICE, adapter, demux); +		printf("using '%s' and '%s'\n", fedev, dmxdev); + +		if ((fefd = open(fedev, O_RDWR | O_NONBLOCK)) < 0) { +			perror("opening frontend failed"); +			return FALSE; +		} +		result = ioctl(fefd, DVBFE_SET_DELSYS, &delivery); +		if (result < 0) { +			perror("ioctl DVBFE_SET_DELSYS failed"); +			close(fefd); +			return FALSE; +		} + +		if ((dmxfdv = open(dmxdev, O_RDWR)) < 0) { +			perror("opening video demux failed"); +			close(fefd); +			return FALSE; +		} + +		if ((dmxfda = open(dmxdev, O_RDWR)) < 0) { +			perror("opening audio demux failed"); +			close(fefd); +			return FALSE; +		} + +		if (dvr == 0)	/* DMX_OUT_DECODER */ +			audiofd = open(auddev, O_RDWR); + +		if (rec_psi){ +			if ((patfd = open(dmxdev, O_RDWR)) < 0) { +				perror("opening pat demux failed"); +				close(audiofd); +				close(dmxfda); +				close(dmxfdv); +				close(fefd); +				return FALSE; +			} + +			if ((pmtfd = open(dmxdev, O_RDWR)) < 0) { +				perror("opening pmt demux failed"); +				close(patfd); +				close(audiofd); +				close(dmxfda); +				close(dmxfdv); +				close(fefd); +				return FALSE; +			} +		} +	} + +	hiband = 0; +	if (lnb_type.switch_val && lnb_type.high_val && +		freq >= lnb_type.switch_val) +		hiband = 1; + +	if (hiband) +		ifreq = freq - lnb_type.high_val; +	else { +		if (freq < lnb_type.low_val) +			ifreq = lnb_type.low_val - freq; +	else +		ifreq = freq - lnb_type.low_val; +	} +	result = FALSE; + +	if (diseqc(fefd, sat_no, pol, hiband)) +		if (do_tune(fefd, ifreq, sr, delsys)) +			if (set_demux(dmxfdv, vpid, DMX_PES_VIDEO, dvr)) +				if (audiofd >= 0) +					(void)ioctl(audiofd, AUDIO_SET_BYPASS_MODE, bypass); +	if (set_demux(dmxfda, apid, DMX_PES_AUDIO, dvr)) { +		if (rec_psi) { +			pmtpid = get_pmt_pid(dmxdev, sid); +			if (pmtpid < 0) { +				result = FALSE; +			} +			if (pmtpid == 0) { +				fprintf(stderr,"couldn't find pmt-pid for sid %04x\n",sid); +				result = FALSE; +			} +			if (set_demux(patfd, 0, DMX_PES_OTHER, dvr)) +				if (set_demux(pmtfd, pmtpid, DMX_PES_OTHER, dvr)) +					result = TRUE; +		} else { +			result = TRUE; +		} +	} + +	check_frontend (fefd, dvr); + +	if (!interactive) { +		close(patfd); +		close(pmtfd); +		if (audiofd >= 0) +			close(audiofd); +		close(dmxfda); +		close(dmxfdv); +		close(fefd); +	} + +	return result; +} + + +static int read_channels(const char *filename, int list_channels, +			 uint32_t chan_no, const char *chan_name, +			 unsigned int adapter, unsigned int frontend, +			 unsigned int demux, int dvr, int rec_psi, +			 int bypass, unsigned int delsys) +{ +	FILE *cfp; +	char buf[4096]; +	char inp[256]; +	char *field, *tmp, *p; +	unsigned int line; +	unsigned int freq, pol, sat_no, sr, vpid, apid, sid; +	int ret; + +again: +	line = 0; +	if (!(cfp = fopen(filename, "r"))) { +		fprintf(stderr, "error opening channel list '%s': %d %m\n", +			filename, errno); +		return FALSE; +	} + +	if (interactive) { +		fprintf(stderr, "\n>>> "); +		if (!fgets(inp, sizeof(inp), stdin)) { +			printf("\n"); +			return -1; +		} +		if (inp[0] == '-' && inp[1] == 'n') { +			chan_no = strtoul(inp+2, NULL, 0); +			chan_name = NULL; +			if (!chan_no) { +				fprintf(stderr, "bad channel number\n"); +				goto again; +			} +		} else { +			p = strchr(inp, '\n'); +			if (p) +			*p = '\0'; +			chan_name = inp; +			chan_no = 0; +		} +	} + +	while (!feof(cfp)) { +		if (fgets(buf, sizeof(buf), cfp)) { +			line++; + +		if (chan_no && chan_no != line) +			continue; + +		tmp = buf; +		field = strsep(&tmp, ":"); + +		if (!field) +			goto syntax_err; + +		if (list_channels) { +			printf("%03u %s\n", line, field); +			continue; +		} + +		if (chan_name && strcasecmp(chan_name, field) != 0) +			continue; + +		printf("zapping to %d '%s':\n", line, field); + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		freq = strtoul(field, NULL, 0); + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		pol = (field[0] == 'h' ? 0 : 1); + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		sat_no = strtoul(field, NULL, 0); + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		sr = strtoul(field, NULL, 0) * 1000; + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		vpid = strtoul(field, NULL, 0); +		if (!vpid) +			vpid = 0x1fff; + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		p = strchr(field, ';'); + +		if (p) { +			*p = '\0'; +			p++; +			if (bypass) { +				if (!p || !*p) +					goto syntax_err; +				field = p; +			} +		} + +		apid = strtoul(field, NULL, 0); +		if (!apid) +			apid = 0x1fff; + +		if (!(field = strsep(&tmp, ":"))) +			goto syntax_err; + +		sid = strtoul(field, NULL, 0); + +		printf("sat %u, frequency = %u MHz %c, symbolrate %u, " +		       "vpid = 0x%04x, apid = 0x%04x sid = 0x%04x\n", +			sat_no, freq, pol ? 'V' : 'H', sr, vpid, apid, sid); + +		fclose(cfp); + +		ret = zap_to(adapter, frontend, demux, sat_no, freq * 1000, +			     pol, sr, vpid, apid, sid, dvr, rec_psi, bypass, delsys); +		if (interactive) +			goto again; + +		if (ret) +			return TRUE; + +		return FALSE; + +syntax_err: +		fprintf(stderr, "syntax error in line %u: '%s'\n", line, buf); +	} else if (ferror(cfp)) { +		fprintf(stderr, "error reading channel list '%s': %d %m\n", +		filename, errno); +		fclose(cfp); +		return FALSE; +	} else +		break; +	} + +	fclose(cfp); + +	if (!list_channels) { +		fprintf(stderr, "channel not found\n"); + +	if (!interactive) +		return FALSE; +	} +	if (interactive) +		goto again; + +	return TRUE; +} + + +void +bad_usage(char *pname, int prlnb) +{ +	int i; +	struct lnb_types_st *lnbp; +	char **cp; + +	if (!prlnb) { +		fprintf (stderr, usage_str, pname); +	} else { +		i = 0; +		fprintf(stderr, "-l <lnb-type> or -l low[,high[,switch]] in Mhz\nwhere <lnb-type> is:\n"); +		while(NULL != (lnbp = lnb_enum(i))) { +			fprintf (stderr, "%s\n", lnbp->name); +			for (cp = lnbp->desc; *cp ; cp++) { +				fprintf (stderr, "   %s\n", *cp); +			} +			i++; +		} +	} +} + +int main(int argc, char *argv[]) +{ +	const char *home; +	char chanfile[2 * PATH_MAX]; +	int list_channels = 0; +	unsigned int chan_no = 0; +	const char *chan_name = NULL; +	unsigned int adapter = 0, frontend = 0, demux = 0, dvr = 0, rec_psi = 0, delsys = 0; +	int bypass = 0; +	int opt, copt = 0; + +	lnb_type = *lnb_enum(0); +	while ((opt = getopt(argc, argv, "hqrpn:a:f:d:t:c:l:xib")) != -1) { +		switch (opt) { +		case '?': +		case 'h': +		default: +			bad_usage(argv[0], 0); +		case 'b': +			bypass = 1; +			break; +		case 'q': +			list_channels = 1; +			break; +		case 'r': +			dvr = 1; +			break; +		case 'n': +			chan_no = strtoul(optarg, NULL, 0); +			break; +		case 'a': +			adapter = strtoul(optarg, NULL, 0); +			break; +		case 'f': +			frontend = strtoul(optarg, NULL, 0); +			break; +		case 'p': +			rec_psi = 1; +			break; +		case 'd': +			demux = strtoul(optarg, NULL, 0); +			break; +		case 't': +			delsys = strtoul(optarg, NULL, 0); +			break; +		case 'c': +			copt = 1; +			strncpy(chanfile, optarg, sizeof(chanfile)); +			break; +		case 'l': +			if (lnb_decode(optarg, &lnb_type) < 0) { +				bad_usage(argv[0], 1); +				return -1; +			} +			break; +		case 'x': +			exit_after_tuning = 1; +			break; +		case 'i': +			interactive = 1; +			exit_after_tuning = 1; +		} +	} +	lnb_type.low_val *= 1000;	/* convert to kiloherz */ +	lnb_type.high_val *= 1000;	/* convert to kiloherz */ +	lnb_type.switch_val *= 1000;	/* convert to kiloherz */ +	if (optind < argc) +		chan_name = argv[optind]; +	if (chan_name && chan_no) { +		bad_usage(argv[0], 0); +		return -1; +	} +	if (list_channels && (chan_name || chan_no)) { +		bad_usage(argv[0], 0); +		return -1; +	} +	if (!list_channels && !chan_name && !chan_no && !interactive) { +		bad_usage(argv[0], 0); +		return -1; +	} + +	if (!copt) { +		if (!(home = getenv("HOME"))) { +			fprintf(stderr, "error: $HOME not set\n"); +		return TRUE; +	} +	snprintf(chanfile, sizeof(chanfile), +		"%s/.szap/%i/%s", home, adapter, CHANNEL_FILE); +	if (access(chanfile, R_OK)) +		snprintf(chanfile, sizeof(chanfile), +			 "%s/.szap/%s", home, CHANNEL_FILE); +	} + +	printf("reading channels from file '%s'\n", chanfile); + +	if (rec_psi) +		dvr=1; + +	if (!read_channels(chanfile, list_channels, chan_no, chan_name, +	    adapter, frontend, demux, dvr, rec_psi, bypass, delsys)) + +		return TRUE; + +	return FALSE; +} diff --git a/test/test.c b/test/test.c index 6c9af51..2881a7a 100644 --- a/test/test.c +++ b/test/test.c @@ -1,4 +1,4 @@ -/*  +/*   * test.c - Test program for new API   *   * Copyright (C) 2000 Ralph  Metzler <ralph@convergence.de> @@ -58,7 +58,7 @@ inline t2a(uint8_t c)    c=reverse[c]&0x7f;    if (c<0x20)      c=0x20; -  +    return c;  } @@ -68,16 +68,16 @@ void testpesfilter(void)  	int i,j;  	int len;  	int fd=open("/dev/ost/demux1", O_RDWR); -	struct dmx_pes_filter_params pesFilterParams;  -	 -	pesFilterParams.input = DMX_IN_FRONTEND;  -	pesFilterParams.output = DMX_OUT_TS_TAP;  -	pesFilterParams.pes_type = DMX_PES_TELETEXT;  +	struct dmx_pes_filter_params pesFilterParams; + +	pesFilterParams.input = DMX_IN_FRONTEND; +	pesFilterParams.output = DMX_OUT_TS_TAP; +	pesFilterParams.pes_type = DMX_PES_TELETEXT;  	pesFilterParams.flags = DMX_IMMEDIATE_START; -   +  	pesFilterParams.pid = 0x2c ;  	if (ioctl(fd, DMX_SET_PES_FILTER, &pesFilterParams) < 0) { -	        printf("Could not set PES filter\n");  +	        printf("Could not set PES filter\n");  		close(fd);  		return;  	} @@ -85,7 +85,7 @@ void testpesfilter(void)  /*  	pesFilterParams.pid = 54;  	if (ioctl(fd, DMX_SET_PES_FILTER, &pesFilterParams) < 0) { -	        printf("Could not set PES filter\n");  +	        printf("Could not set PES filter\n");  		close(fd);  		return;  	} @@ -93,7 +93,7 @@ void testpesfilter(void)  	pesFilterParams.pid = 55;  	if (ioctl(fd, DMX_SET_PES_FILTER, &pesFilterParams) < 0) { -	        printf("Could not set PES filter\n");  +	        printf("Could not set PES filter\n");  		close(fd);  		return;  	} @@ -103,7 +103,7 @@ void testpesfilter(void)  		if (len>0) write(1, buf, len);  	} -	do {  +	do {  	        read(fd, buf, 4);  		if (htonl(*(uint32_t *)buf)!=0x00001bd)  		        continue; @@ -112,10 +112,10 @@ void testpesfilter(void)  	        read(fd, buf+6, len);  		fprintf(stderr,"read %d bytes PES\n", len);  		write (1, buf, len+6); -		 +  		id=buf[45];  		fprintf(stderr,"id=%02x\n", id); -		 +  		for (i=0; i<(len+6-46)/46; i++) {  		  for (j=6; j<46; j++) {  		    fprintf(stderr,"%c", t2a(buf[i*46+46+j])); @@ -128,11 +128,11 @@ void testpesfilter(void)  	/*  	pesFilterParams.pid = 55; -	pesFilterParams.input = DMX_IN_FRONTEND;  -	pesFilterParams.output = DMX_OUT_DECODER;  -	pesFilterParams.pes_type = DMX_PES_TELETEXT;  +	pesFilterParams.input = DMX_IN_FRONTEND; +	pesFilterParams.output = DMX_OUT_DECODER; +	pesFilterParams.pes_type = DMX_PES_TELETEXT;  	pesFilterParams.flags = DMX_IMMEDIATE_START; -   +  	ioctl(fd, DMX_SET_PES_FILTER, &pesFilterParams);  	close(fd);  	*/ @@ -145,8 +145,8 @@ senf()    int len;    struct secCommand scmd;    struct secCmdSequence scmds; -  struct dmx_pes_filter_params pesFilterParams;  -  struct dmx_sct_filter_params secFilterParams;  +  struct dmx_pes_filter_params pesFilterParams; +  struct dmx_sct_filter_params secFilterParams;    FrontendParameters frp;    uint8_t buf[4096]; @@ -173,7 +173,7 @@ senf()    scmd.u.diseqc.cmd=0x38;    scmd.u.diseqc.numParams=1;    scmd.u.diseqc.params[0]=0xf0; -   +    scmds.voltage=SEC_VOLTAGE_13;    scmds.miniCommand=SEC_MINI_NONE;    scmds.continuousTone=SEC_TONE_ON; @@ -195,31 +195,31 @@ senf()      if (ret<0)        perror("DMX_SET_BUFFER_SIZE\n");    printf("Audio filter size OK\n"); -  pesFilterParams.pid = 0x60;  -  pesFilterParams.input = DMX_IN_FRONTEND;  -  pesFilterParams.output = DMX_OUT_DECODER;  -  pesFilterParams.pes_type = DMX_PES_AUDIO;  +  pesFilterParams.pid = 0x60; +  pesFilterParams.input = DMX_IN_FRONTEND; +  pesFilterParams.output = DMX_OUT_DECODER; +  pesFilterParams.pes_type = DMX_PES_AUDIO;    pesFilterParams.flags = DMX_IMMEDIATE_START; -   -  if (ioctl(fd_demux2, DMX_SET_PES_FILTER, &pesFilterParams) < 0)   return(1);  + +  if (ioctl(fd_demux2, DMX_SET_PES_FILTER, &pesFilterParams) < 0)   return(1);    printf("Audio filter OK\n"); -   -  if (ioctl(fd_demux, DMX_SET_BUFFER_SIZE, 64*1024) < 0)  return(1);  + +  if (ioctl(fd_demux, DMX_SET_BUFFER_SIZE, 64*1024) < 0)  return(1);    pesFilterParams.pid = 0xa2; -  pesFilterParams.input = DMX_IN_FRONTEND;  -  pesFilterParams.output = DMX_OUT_DECODER;  -  pesFilterParams.pes_type = DMX_PES_VIDEO;  +  pesFilterParams.input = DMX_IN_FRONTEND; +  pesFilterParams.output = DMX_OUT_DECODER; +  pesFilterParams.pes_type = DMX_PES_VIDEO;    pesFilterParams.flags = DMX_IMMEDIATE_START; -  if (ioctl(fd_demux, DMX_SET_PES_FILTER, &pesFilterParams) < 0)  return(1);  +  if (ioctl(fd_demux, DMX_SET_PES_FILTER, &pesFilterParams) < 0)  return(1);    printf("Video filter OK\n");  #endif    /*    pesFilterParams.pid = 56; -  pesFilterParams.input = DMX_IN_FRONTEND;  -  pesFilterParams.output = DMX_OUT_DECODER;  -  pesFilterParams.pes_type = DMX_PES_TELETEXT;  +  pesFilterParams.input = DMX_IN_FRONTEND; +  pesFilterParams.output = DMX_OUT_DECODER; +  pesFilterParams.pes_type = DMX_PES_TELETEXT;    pesFilterParams.flags = DMX_IMMEDIATE_START; -  if (ioctl(fd_tt, DMX_SET_PES_FILTER, &pesFilterParams) < 0)  return(1);  +  if (ioctl(fd_tt, DMX_SET_PES_FILTER, &pesFilterParams) < 0)  return(1);    printf("TT filter OK\n");    */    //while (1); @@ -250,7 +250,7 @@ senf()      int32_t snr, str;      ioctl(fd_frontend, FE_READ_SNR, &snr);      ioctl(fd_frontend, FE_READ_SIGNAL_STRENGTH, &str); -     +      printf("snr=%d, str=%d\n", snr, str);    }    */ @@ -260,7 +260,7 @@ senf()    if (ioctl(fd_section, DMX_SET_FILTER, &secFilterParams) < 0)  return;    //if (ioctl(fd_section2, DMX_SET_FILTER, &secFilterParams) < 0)  return;    //close(fd_section2); -  //while (1)  +  //while (1)  {      len=read(fd_section, buf, 4096);      if (len>0) write(1, buf, len); @@ -270,7 +270,7 @@ senf()      //if (len>0) write(1,buf,len);      //printf("read section with length %d\n", len);    } -   +  }  main() @@ -278,4 +278,3 @@ main()  	//senf();    testpesfilter();  } - diff --git a/test/test_audio.c b/test/test_audio.c index 0abb734..f7bfacc 100644 --- a/test/test_audio.c +++ b/test/test_audio.c @@ -1,4 +1,4 @@ -/*  +/*   * test_audio.c - Test program for new API   *   * Copyright (C) 2000 Ralph  Metzler <ralph@convergence.de> @@ -40,7 +40,7 @@ int audioStop(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_STOP,0) < 0)){ +	if ((ans = ioctl(fd,AUDIO_STOP,0)) < 0) {  		perror("AUDIO STOP: ");  		return -1;  	} @@ -52,7 +52,7 @@ int audioPlay(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_PLAY) < 0)){ +	if ((ans = ioctl(fd,AUDIO_PLAY)) < 0) {  		perror("AUDIO PLAY: ");  		return -1;  	} @@ -65,7 +65,7 @@ int audioPause(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_PAUSE) < 0)){ +	if ((ans = ioctl(fd,AUDIO_PAUSE)) < 0) {  		perror("AUDIO PAUSE: ");  		return -1;  	} @@ -78,7 +78,7 @@ int audioContinue(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_CONTINUE) < 0)){ +	if ((ans = ioctl(fd,AUDIO_CONTINUE)) < 0) {  		perror("AUDIO CONTINUE: ");  		return -1;  	} @@ -90,7 +90,7 @@ int audioSelectSource(int fd, audio_stream_source_t source)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SELECT_SOURCE, source) < 0)){ +	if ((ans = ioctl(fd,AUDIO_SELECT_SOURCE, source)) < 0) {  		perror("AUDIO SELECT SOURCE: ");  		return -1;  	} @@ -104,7 +104,7 @@ int audioSetMute(int fd, boolean state)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SET_MUTE, state) < 0)){ +	if ((ans = ioctl(fd,AUDIO_SET_MUTE, state)) < 0) {  		perror("AUDIO SET MUTE: ");  		return -1;  	} @@ -116,7 +116,7 @@ int audioSetAVSync(int fd,boolean state)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SET_AV_SYNC, state) < 0)){ +	if ((ans = ioctl(fd,AUDIO_SET_AV_SYNC, state)) < 0) {  		perror("AUDIO SET AV SYNC: ");  		return -1;  	} @@ -128,8 +128,8 @@ int audioSetBypassMode(int fd,boolean mode)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SET_BYPASS_MODE, mode) < 0)){ -		perror("AUDIO SET BYPASS MODE: "); +	if ((ans = ioctl(fd,AUDIO_SET_BYPASS_MODE, mode)) < 0) { +		printf("AUDIO SET BYPASS MODE not implemented?\n");  		return -1;  	} @@ -141,7 +141,7 @@ int audioChannelSelect(int fd, audio_channel_select_t select)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_CHANNEL_SELECT, select) < 0)){ +	if ((ans = ioctl(fd,AUDIO_CHANNEL_SELECT, select)) < 0) {  		perror("AUDIO CHANNEL SELECT: ");  		return -1;  	} @@ -154,7 +154,7 @@ int audioGetStatus(int fd)  	struct audio_status stat;  	int ans; -	if ( (ans = ioctl(fd,AUDIO_GET_STATUS, &stat) < 0)){ +	if ((ans = ioctl(fd,AUDIO_GET_STATUS, &stat)) < 0) {  		perror("AUDIO GET STATUS: ");  		return -1;  	} @@ -179,7 +179,7 @@ int audioGetStatus(int fd)  		printf("unknown (%d)\n",stat.play_state);  		break;  	} -	 +  	printf("  Stream Source       : ");  	switch((int)stat.stream_source){  	case AUDIO_SOURCE_DEMUX: @@ -227,7 +227,7 @@ play_file_audio(int filefd, int fd)  	int stopped = 0;  	boolean mute = false;  	boolean sync = false; -	 +  	pfd[0].fd = STDIN_FILENO;  	pfd[0].events = POLLIN; @@ -257,7 +257,7 @@ play_file_audio(int filefd, int fd)  						printf("playback stopped\n");  						stopped = 1;  						break; -						 +  					case 'c':  						audioContinue(fd);  						printf("playback continued\n"); @@ -291,7 +291,7 @@ play_file_audio(int filefd, int fd)  						break;  					}  				} -				 +  			}  		}  	} @@ -312,8 +312,8 @@ main(int argc, char **argv)  		perror("File open:");  		return -1;  	} -	     -	if((fd = open("/dev/ost/audio",O_RDWR|O_NONBLOCK)) < 0){ + +	if ((fd = open("/dev/ost/audio",O_RDWR|O_NONBLOCK)) < 0){  		perror("AUDIO DEVICE: ");  		return -1;  	} @@ -321,7 +321,7 @@ main(int argc, char **argv)  	audioSetMute(fd,mute); -	//	audioSetBypassMode(fd,false); // not implemented +	audioSetBypassMode(fd,false);  	//audioContinue(fd);  	audioSelectSource(fd,AUDIO_SOURCE_MEMORY);  	audioPlay(fd); @@ -334,7 +334,7 @@ main(int argc, char **argv)  	//audioChannelSelect(fd,AUDIO_STEREO);  	//audioSetAVSync(fd,sync);  	audioGetStatus(fd); -	 +  	play_file_audio(filefd,fd);  	close(fd); @@ -342,4 +342,3 @@ main(int argc, char **argv)  } - diff --git a/test/test_av.c b/test/test_av.c index 5d77000..6c32848 100644 --- a/test/test_av.c +++ b/test/test_av.c @@ -1,4 +1,4 @@ -/*  +/*   * test_av.c - Test for audio and video MPEG decoder API.   *   * Copyright (C) 2000 - 2002 convergence GmbH @@ -30,6 +30,7 @@  #include <time.h>  #include <unistd.h> +#include <linux/types.h>  #include <linux/dvb/audio.h>  #include <linux/dvb/video.h> @@ -94,11 +95,11 @@ int audioSetMute(int fd, char *arg)  int audioSetAVSync(int fd, char *arg)  { -	int sync; +	int _sync;  	if (!arg)  		return -1; -	sync = atoi(arg); -	if (ioctl(fd, AUDIO_SET_AV_SYNC, sync) == -1) +	_sync = atoi(arg); +	if (ioctl(fd, AUDIO_SET_AV_SYNC, _sync) == -1)  		perror("AUDIO_SET_AV_SYNC");  	return 0;  } @@ -127,66 +128,66 @@ int audioChannelSelect(int fd, char *arg)  int audioGetStatus(int fd, char *arg)  { -	struct audio_status stat; +	struct audio_status _stat;  	if (arg)  		return -1; -	if (ioctl(fd, AUDIO_GET_STATUS, &stat) == -1) { +	if (ioctl(fd, AUDIO_GET_STATUS, &_stat) == -1) {  		perror("AUDIO_GET_STATUS");  		return 0;  	}  	printf("Audio Status:\n");  	printf("  Sync State          : %s\n", -	       (stat.AV_sync_state ? "SYNC" : "NO SYNC")); +	       (_stat.AV_sync_state ? "SYNC" : "NO SYNC"));  	printf("  Mute State          : %s\n", -	       (stat.mute_state ? "muted" : "not muted")); +	       (_stat.mute_state ? "muted" : "not muted"));  	printf("  Play State          : "); -	switch ((int)stat.play_state){ +	switch ((int)_stat.play_state){  	case AUDIO_STOPPED: -		printf("STOPPED (%d)\n",stat.play_state); +		printf("STOPPED (%d)\n",_stat.play_state);  		break;  	case AUDIO_PLAYING: -		printf("PLAYING (%d)\n",stat.play_state); +		printf("PLAYING (%d)\n",_stat.play_state);  		break;  	case AUDIO_PAUSED: -		printf("PAUSED (%d)\n",stat.play_state); +		printf("PAUSED (%d)\n",_stat.play_state);  		break;  	default: -		printf("unknown (%d)\n",stat.play_state); +		printf("unknown (%d)\n",_stat.play_state);  		break;  	}  	printf("  Stream Source       : "); -	switch((int)stat.stream_source){ +	switch((int)_stat.stream_source){  	case AUDIO_SOURCE_DEMUX: -		printf("DEMUX (%d)\n",stat.stream_source); +		printf("DEMUX (%d)\n",_stat.stream_source);  		break;  	case AUDIO_SOURCE_MEMORY: -		printf("MEMORY (%d)\n",stat.stream_source); +		printf("MEMORY (%d)\n",_stat.stream_source);  		break;  	default: -		printf("unknown (%d)\n",stat.stream_source); +		printf("unknown (%d)\n",_stat.stream_source);  		break;  	}  	printf("  Channel Select      : "); -	switch((int)stat.channel_select){ +	switch((int)_stat.channel_select){  	case AUDIO_STEREO: -		printf("Stereo (%d)\n",stat.channel_select); +		printf("Stereo (%d)\n",_stat.channel_select);  		break;  	case AUDIO_MONO_LEFT: -		printf("Mono left(%d)\n",stat.channel_select); +		printf("Mono left(%d)\n",_stat.channel_select);  		break;  	case AUDIO_MONO_RIGHT: -		printf("Mono right (%d)\n",stat.channel_select); +		printf("Mono right (%d)\n",_stat.channel_select);  		break;  	default: -		printf("unknown (%d)\n",stat.channel_select); +		printf("unknown (%d)\n",_stat.channel_select);  		break;  	}  	printf("  Bypass Mode         : %s\n", -	       (stat.bypass_mode ? "ON" : "OFF")); +	       (_stat.bypass_mode ? "ON" : "OFF"));  	return 0; @@ -300,76 +301,76 @@ int videoSlowMotion(int fd, char *arg)  int videoGetStatus(int fd, char *arg)  { -	struct video_status stat; +	struct video_status _stat;  	if (arg)  		return -1; -	if (ioctl(fd, VIDEO_GET_STATUS, &stat) == -1){ +	if (ioctl(fd, VIDEO_GET_STATUS, &_stat) == -1){  		perror("VIDEO_GET_STATUS");  		return 0;  	}  	printf("Video Status:\n");  	printf("  Blank State          : %s\n", -	       (stat.video_blank ? "BLANK" : "STILL")); +	       (_stat.video_blank ? "BLANK" : "STILL"));  	printf("  Play State           : "); -	switch ((int)stat.play_state){ +	switch ((int)_stat.play_state){  	case VIDEO_STOPPED: -		printf("STOPPED (%d)\n",stat.play_state); +		printf("STOPPED (%d)\n",_stat.play_state);  		break;  	case VIDEO_PLAYING: -		printf("PLAYING (%d)\n",stat.play_state); +		printf("PLAYING (%d)\n",_stat.play_state);  		break;  	case VIDEO_FREEZED: -		printf("FREEZED (%d)\n",stat.play_state); +		printf("FREEZED (%d)\n",_stat.play_state);  		break;  	default: -		printf("unknown (%d)\n",stat.play_state); +		printf("unknown (%d)\n",_stat.play_state);  		break;  	}  	printf("  Stream Source        : "); -	switch((int)stat.stream_source){ +	switch((int)_stat.stream_source){  	case VIDEO_SOURCE_DEMUX: -		printf("DEMUX (%d)\n",stat.stream_source); +		printf("DEMUX (%d)\n",_stat.stream_source);  		break;  	case VIDEO_SOURCE_MEMORY: -		printf("MEMORY (%d)\n",stat.stream_source); +		printf("MEMORY (%d)\n",_stat.stream_source);  		break;  	default: -		printf("unknown (%d)\n",stat.stream_source); +		printf("unknown (%d)\n",_stat.stream_source);  		break;  	}  	printf("  Format (Aspect Ratio): "); -	switch((int)stat.video_format){ +	switch((int)_stat.video_format){  	case VIDEO_FORMAT_4_3: -		printf("4:3 (%d)\n",stat.video_format); +		printf("4:3 (%d)\n",_stat.video_format);  		break;  	case VIDEO_FORMAT_16_9: -		printf("16:9 (%d)\n",stat.video_format); +		printf("16:9 (%d)\n",_stat.video_format);  		break;  	case VIDEO_FORMAT_221_1: -		printf("2.21:1 (%d)\n",stat.video_format); +		printf("2.21:1 (%d)\n",_stat.video_format);  		break;  	default: -		printf("unknown (%d)\n",stat.video_format); +		printf("unknown (%d)\n",_stat.video_format);  		break;  	}  	printf("  Display Format       : "); -	switch((int)stat.display_format){ +	switch((int)_stat.display_format){  	case VIDEO_PAN_SCAN: -		printf("Pan&Scan (%d)\n",stat.display_format); +		printf("Pan&Scan (%d)\n",_stat.display_format);  		break;  	case VIDEO_LETTER_BOX: -		printf("Letterbox (%d)\n",stat.display_format); +		printf("Letterbox (%d)\n",_stat.display_format);  		break;  	case VIDEO_CENTER_CUT_OUT: -		printf("Center cutout (%d)\n",stat.display_format); +		printf("Center cutout (%d)\n",_stat.display_format);  		break;  	default: -		printf("unknown (%d)\n",stat.display_format); +		printf("unknown (%d)\n",_stat.display_format);  		break;  	}  	return 0; @@ -424,12 +425,12 @@ int videoStillPicture(int fd, char *arg)  	sp.size = st.st_size;  	printf("I-frame size: %d\n", sp.size); -	if(!sp.iFrame) { +	if (!sp.iFrame) {  		printf("No memory for I-Frame\n");  		return 0;  	} -	printf("read: %d bytes\n",read(sifd,sp.iFrame,sp.size)); +	printf("read: %d bytes\n", (int) read(sifd,sp.iFrame,sp.size));  	if (ioctl(fd, VIDEO_STILLPICTURE, &sp) == -1)  		perror("VIDEO_STILLPICTURE");  	return 0; @@ -555,11 +556,11 @@ int main(void)  	printf("using audio device '%s'\n", audiodev);  	printf("enter command or h + enter for help\n"); -	if((vfd = open(videodev, O_RDWR | O_NONBLOCK)) < 0) { +	if ((vfd = open(videodev, O_RDWR | O_NONBLOCK)) < 0) {  		perror("open video device");  		return 1;  	} -	if((afd = open(audiodev, O_RDWR | O_NONBLOCK)) < 0) { +	if ((afd = open(audiodev, O_RDWR | O_NONBLOCK)) < 0) {  		perror("open audio device");  		return 1;  	} @@ -571,4 +572,3 @@ int main(void)  	close(afd);  	return 0;  } - diff --git a/test/test_av_play.c b/test/test_av_play.c index 293d6d8..b7ad867 100644 --- a/test/test_av_play.c +++ b/test/test_av_play.c @@ -19,27 +19,44 @@   * along with this program; if not, write to the Free Software   * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.   * + * Thu Jun 24 09:18:44 CEST 2004 + *   Add scan_file_av() and copy_to_dvb() for AV + *   filtering to be able to use AC3 audio streams + *   Copyright (C) 2004 Werner Fink <werner@suse.de>   */ +#ifndef _GNU_SOURCE +# define _GNU_SOURCE +#endif +  #include <sys/ioctl.h>  #include <stdio.h> +#include <string.h>  #include <stdlib.h>  #include <sys/types.h>  #include <sys/stat.h> +#include <netinet/in.h>  #include <fcntl.h>  #include <time.h> +#include <termios.h>  #include <unistd.h> +#include <errno.h>  #include <linux/dvb/dmx.h>  #include <linux/dvb/video.h>  #include <linux/dvb/audio.h>  #include <sys/poll.h> +static char dolby; +static char audio; +static char black; +static char volset; +  static int audioPlay(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_PLAY) < 0)){ +	if ((ans = ioctl(fd,AUDIO_PLAY)) < 0) {  		perror("AUDIO PLAY: ");  		return -1;  	} @@ -47,12 +64,11 @@ static int audioPlay(int fd)  	return 0;  } -  static int audioSelectSource(int fd, audio_stream_source_t source)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SELECT_SOURCE, source) < 0)){ +	if ((ans = ioctl(fd,AUDIO_SELECT_SOURCE, source)) < 0) {  		perror("AUDIO SELECT SOURCE: ");  		return -1;  	} @@ -60,13 +76,11 @@ static int audioSelectSource(int fd, audio_stream_source_t source)  	return 0;  } - -  static int audioSetMute(int fd, int state)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SET_MUTE, state) < 0)){ +	if ((ans = ioctl(fd,AUDIO_SET_MUTE, state)) < 0) {  		perror("AUDIO SET MUTE: ");  		return -1;  	} @@ -78,7 +92,7 @@ static int audioSetAVSync(int fd, int state)  {  	int ans; -	if ( (ans = ioctl(fd,AUDIO_SET_AV_SYNC, state) < 0)){ +	if ((ans = ioctl(fd,AUDIO_SET_AV_SYNC, state)) < 0) {  		perror("AUDIO SET AV SYNC: ");  		return -1;  	} @@ -86,11 +100,55 @@ static int audioSetAVSync(int fd, int state)  	return 0;  } +static int audioSetVolume(int fd, int level) +{ +	int ans; +	audio_mixer_t mix; + +	mix.volume_left = mix.volume_right = level; +	if ((ans = ioctl(fd, AUDIO_SET_MIXER, &mix)) < 0) { +		perror("AUDIO SET VOLUME: "); +		return -1; +	} +	return 0; +} + +static int audioStop(int fd) +{ +	int ans; + +	if ((ans = ioctl(fd,AUDIO_STOP,0)) < 0) { +		perror("AUDIO STOP: "); +		return -1; +	} +	return 0; +} + +static int deviceClear(int afd, int vfd) +{ +	int ans; + +	if (vfd >= 0) { +		if ((ans = ioctl(vfd, VIDEO_CLEAR_BUFFER, 0)) < 0) { +			perror("VIDEO CLEAR BUFFER: "); +			return -1; +		} +	} +	if (afd >= 0) { +		if ((ans = ioctl(afd, AUDIO_CLEAR_BUFFER, 0)) < 0) { +			perror("AUDIO CLEAR BUFFER: "); +			return -1; +		} +	} + +	return 0; +} +  static int videoStop(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_STOP,0) < 0)){ +	if ((ans = ioctl(fd,VIDEO_STOP,0)) < 0) {  		perror("VIDEO STOP: ");  		return -1;  	} @@ -102,7 +160,7 @@ static int videoPlay(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_PLAY) < 0)){ +	if ((ans = ioctl(fd,VIDEO_PLAY)) < 0) {  		perror("VIDEO PLAY: ");  		return -1;  	} @@ -110,12 +168,11 @@ static int videoPlay(int fd)  	return 0;  } -  static int videoFreeze(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_FREEZE) < 0)){ +	if ((ans = ioctl(fd,VIDEO_FREEZE)) < 0) {  		perror("VIDEO FREEZE: ");  		return -1;  	} @@ -123,12 +180,22 @@ static int videoFreeze(int fd)  	return 0;  } +static int videoBlank(int fd, int state) +{ +	int ans; + +	if ((ans = ioctl(fd, VIDEO_SET_BLANK, state)) < 0) { +		perror("VIDEO BLANK: "); +		return -1; +	} +	return 0; +}  static int videoContinue(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_CONTINUE) < 0)){ +	if ((ans = ioctl(fd,VIDEO_CONTINUE)) < 0) {  		perror("VIDEO CONTINUE: ");  		return -1;  	} @@ -140,7 +207,7 @@ static int videoSelectSource(int fd, video_stream_source_t source)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_SELECT_SOURCE, source) < 0)){ +	if ((ans = ioctl(fd,VIDEO_SELECT_SOURCE, source)) < 0) {  		perror("VIDEO SELECT SOURCE: ");  		return -1;  	} @@ -153,7 +220,7 @@ static int videoFastForward(int fd,int nframes)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_FAST_FORWARD, nframes) < 0)){ +	if ((ans = ioctl(fd,VIDEO_FAST_FORWARD, nframes)) < 0) {  		perror("VIDEO FAST FORWARD: ");  		return -1;  	} @@ -165,7 +232,7 @@ static int videoSlowMotion(int fd,int nframes)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_SLOWMOTION, nframes) < 0)){ +	if ((ans = ioctl(fd,VIDEO_SLOWMOTION, nframes)) < 0) {  		perror("VIDEO SLOWMOTION: ");  		return -1;  	} @@ -173,14 +240,11 @@ static int videoSlowMotion(int fd,int nframes)  	return 0;  } -#define BUFFY 32768 -#define NFD   2 -static void play_file_av(int filefd, int vfd, int afd) +#define NFD   3 +static void copy_to_dvb(int vfd, int afd, int cfd, const uint8_t* ptr, const unsigned short len)  { -	char buf[BUFFY]; -	int count; -	int written;  	struct pollfd pfd[NFD]; +	unsigned short pos = 0;  	int stopped = 0;  	pfd[0].fd = STDIN_FILENO; @@ -192,119 +256,399 @@ static void play_file_av(int filefd, int vfd, int afd)  	pfd[2].fd = afd;  	pfd[2].events = POLLOUT; -	videoSelectSource(vfd,VIDEO_SOURCE_MEMORY); -	audioSelectSource(afd,AUDIO_SOURCE_MEMORY); - -	// FIXME: only seems to work if starting audio first! -	audioPlay(afd); -	videoPlay(vfd); - -	count = read(filefd,buf,BUFFY); -	write(vfd,buf,count); - -	while ( (count = read(filefd,buf,BUFFY)) >= 0  ){ -		written = 0; -		while(written < count){ -			if (poll(pfd,NFD,1)){ -				if (pfd[1].revents & POLLOUT){ -					written += write(vfd,buf+written, -							count-written); +	while (pos < len) { +		int ret; +		if ((ret = poll(pfd,NFD,1)) > 0) { +			if (pfd[1].revents & POLLOUT) { +				int cnt = write(cfd, ptr + pos, len - pos); +				if (cnt > 0) +					pos += cnt; +				else if (cnt < 0) { +					if (errno != EAGAIN && errno != EINTR) { +						perror("Write:"); +						exit(-1); +					} +					if (errno == EAGAIN) +						usleep(1000); +					continue;  				} -				if (pfd[0].revents & POLLIN){ -					int c = getchar(); -					switch(c){ -					case 'z': +			} +			if (pfd[0].revents & POLLIN) { +				int c = getchar(); +				switch(c) { +				case 'z': +					if (audio && !black) { +						audioSetMute(afd, 1); +					} else {  						videoFreeze(vfd); -						printf("playback frozen\n"); -						stopped = 1; -						break; - -					case 's': +					} +					deviceClear(afd, -1); +					printf("playback frozen\n"); +					stopped = 1; +					break; + +				case 's': +					if (audio) { +						audioStop(afd); +						deviceClear(afd, -1); +					} else {  						videoStop(vfd); -						printf("playback stopped\n"); -						stopped = 1; -						break; - -					case 'c': +						deviceClear(afd, vfd); +					} +					printf("playback stopped\n"); +					stopped = 1; +					break; + +				case 'c': +					if (audio && !black) { +						audioSetAVSync(afd, 0); +						deviceClear(afd, -1); +						audioSetMute(afd, 0); +					} else { +						audioSetAVSync(afd, 1); +						deviceClear(afd, vfd);  						videoContinue(vfd); -						printf("playback continued\n"); -						stopped = 0; -						break; - -					case 'p': -						videoPlay(vfd); +					} +					printf("playback continued\n"); +					stopped = 0; +					break; + +				case 'p': +					if (audio) { +						deviceClear(afd, -1); +						audioSetAVSync(afd, 0);  						audioPlay(afd); -					        audioSetAVSync(afd, 1); -						audioSetMute(afd, 0); -						printf("playback started\n"); -						stopped = 0; -						break; - -					case 'f': -					        audioSetAVSync(afd, 0); +					} else { +						deviceClear(afd, vfd); +						audioSetAVSync(afd, 1); +						audioPlay(afd); +						videoPlay(vfd); +					} +					audioSetMute(afd, 0); +					printf("playback started\n"); +					stopped = 0; +					break; + +				case 'f': +					audioSetAVSync(afd, 0); +					if (!audio) {  						audioSetMute(afd, 1);  						videoFastForward(vfd,0); -						printf("fastforward\n"); -						stopped = 0; -						break; - -					case 'm': -					        audioSetAVSync(afd, 0); +					} +					printf("fastforward\n"); +					stopped = 0; +					break; + +				case 'm': +					audioSetAVSync(afd, 0); +					audioSetMute(afd, 1); +					printf("mute\n"); +					stopped = 0; +					break; + +				case 'u': +					audioSetAVSync(afd, 1); +					audioSetMute(afd, 0); +					printf("unmute\n"); +					stopped = 0; +					break; + +				case 'd': +					if (dolby) +						dolby = 0; +					else +						dolby++; +					break; + +				case 'l': +					audioSetAVSync(afd, 0); +					if (!audio) {  						audioSetMute(afd, 1);  						videoSlowMotion(vfd,2); -						printf("slowmotion\n"); -						stopped = 0; -						break; +					} +					printf("slowmotion\n"); +					stopped = 0; +					break; -					case 'q': -						videoContinue(vfd); -						exit(0); +				case 'q': +					videoContinue(vfd); +					exit(0); +					break; + +				default: +					break; +				} +			} +		} else if (ret < 0) { +			if (errno != EAGAIN && errno != EINTR) { +				perror("Write:"); +				exit(-1); +			} +			if (errno == EAGAIN) +				usleep(1000); +		} +	} +} + +static unsigned char play[6] = {0x00, 0x00, 0x01, 0xff, 0xff, 0xff}; +static unsigned char except[2]; + +static int scan_file_av(int vfd, int afd, const unsigned char *buf, int buflen) +{ +	const unsigned char *const start = buf; +	const unsigned char *const   end = buf + buflen; + +	static unsigned int magic = 0xffffffff; +	static unsigned short count, len; +	static int fdc = -1; +	int m; + +	while (buf < end) { +		if (count < 6) { +			switch (count) { +			case 0: +				m = 0; +				while ((magic & 0xffffff00) != 0x00000100) { +					if (buf >= end) goto out; +					magic = (magic << 8) | *buf++; +					m++; +				} +				if (m > 4) +					printf("Broken Frame found\n"); +				play[3] = (unsigned char)(magic & 0x000000ff); +				switch (play[3]) { +				case 0xE0 ... 0xEF: +					fdc = vfd; +					if (except[0] != play[3]) { +						if (except[0] == 0) +							except[0] = play[3]; +						else +							fdc = -1; +					} +					if (audio) +						fdc = -1; +					break; +				case 0xC0 ... 0xDF: +					fdc = afd; +					if (dolby) { +						fdc = -1;  						break;  					} +					if (except[1] != play[3]) { +						if (except[1] == 0) +							except[1] = play[3]; +						else +							fdc = -1; +					} +					if (!volset) { +						audioSetVolume(afd, 255); +						volset = 1; +					} +					break; +				case 0xBD: +					/* +					 * TODO: sub filter to through out e.g. ub pictures +					 * in Private Streams 1 _and_ get sub audio header +					 * to set an except(ion) audio stream. +					 * The later one requires some changes within the VDR +					 * remux part!  2004/07/01 Werner +					 */ +					fdc = afd; +					if (!dolby) { +						fdc = -1; +						break; +					} +					if (volset) { +						audioSetVolume(afd, 0); +						volset = 0; +					} +					break; +				default: +					fdc = -1; +					break;  				} +				count = 4; +			case 4: +				if (buf >= end) goto out; +				len = ((*buf) << 8); +				play[4] = (*buf); +				buf++; +				count++; +			case 5: +				if (buf >= end) goto out; +				len |= (*buf); +				len += 6; +				play[5] = (*buf); +				buf++; +				count++; +				if (fdc != -1) +					copy_to_dvb(vfd, afd, fdc, &play[0], count); +			default: +				break;  			}  		} + +		while (count < len) { +			int rest = end - buf; +			if (rest <= 0) goto out; + +			if (rest + count > len) +				rest = len - count; + +			if (fdc != -1) +				copy_to_dvb(vfd, afd, fdc, buf, rest); +			count += rest; +			buf   += rest; +		} + +		/* Reset for next scan */ +		magic = 0xffffffff; +		count = len = 0; +		fdc = -1; +		play[3] = 0xff;  	} +out: +	return buf - start; +} + +#define BUFFY 32768 +static void play_file_av(int filefd, int vfd, int afd) +{ +	unsigned char buf[BUFFY]; +	int count; + +	audioSetMute(afd, 1); +	videoBlank(vfd, 1); +	if (audio && !black) { +		audioStop(afd); +		deviceClear(afd, -1); +		audioSetAVSync(afd, 0); +		audioSelectSource(afd, AUDIO_SOURCE_MEMORY); +		audioPlay(afd); +		videoBlank(vfd, 0); +	} else if (audio && black) { +		deviceClear(afd, vfd); +		videoBlank(vfd, 1); +		audioSetAVSync(afd, 0); +		audioSelectSource(afd, AUDIO_SOURCE_MEMORY); +		videoSelectSource(vfd, VIDEO_SOURCE_MEMORY); +		audioPlay(afd); +		videoPlay(vfd); +	} else { +		deviceClear(afd, vfd); +		audioSetAVSync(afd, 1); +		audioSelectSource(afd, AUDIO_SOURCE_MEMORY); +		videoSelectSource(vfd, VIDEO_SOURCE_MEMORY); +		audioPlay(afd); +		videoPlay(vfd); +		videoBlank(vfd, 0); +	} + +	if (dolby) { +		audioSetVolume(afd, 0); +		volset = 0; +	} else { +		audioSetVolume(afd, 255); +		volset = 1; +	} + +#ifndef __stub_posix_fadvise +	posix_fadvise(filefd, 0, 0, POSIX_FADV_SEQUENTIAL); +#endif + +	while ((count = read(filefd,buf,BUFFY)) > 0) +		scan_file_av(vfd,afd,buf,count); +} + +static struct termios term; + +static void restore(void) +{ +	tcsetattr(STDIN_FILENO, TCSANOW, &term);  }  int main(int argc, char **argv)  { -	int vfd, afd; +	int vfd, afd, c;  	int filefd; -	char *videodev = "/dev/dvb/adapter0/video0"; -	char *audiodev = "/dev/dvb/adapter0/audio0"; +	const char *videodev = "/dev/dvb/adapter0/video0"; +	const char *audiodev = "/dev/dvb/adapter0/audio0"; + +	if (((tcgetpgrp(STDIN_FILENO) == getpid()) || (getppid() != (pid_t)1)) +	    && (tcgetattr(STDIN_FILENO, &term) == 0)) { +		struct termios newterm; +		memcpy(&newterm, &term, sizeof(struct termios)); +		newterm.c_iflag = 0; +		newterm.c_lflag &= ~(ICANON | ECHO); +		newterm.c_cc[VMIN] = 0; +		newterm.c_cc[VTIME] = 0; +		atexit(restore); +		tcsetattr(STDIN_FILENO, TCSANOW, &newterm); +	} -	if (argc < 2) { -		fprintf(stderr, "usage: test_av_play mpeg_A+V_PES_file\n"); -		return 1; +	opterr = 0; +	while ((c = getopt(argc, argv, "+daA")) != -1) { +		switch (c) { +		case 'd': +			dolby++; +			break; +		case 'a': +			audio++; +			break; +		case 'A': +			audio++; +			black++; +			break; +		case '?': +			fprintf(stderr, "usage: test_av_play [-d] [-a] [-A] mpeg_A+V_PES_file\n"); +			return 1; +		default: +			break; +		}  	} +	argv += optind; +	argc -= optind;  	if (getenv("VIDEO"))  		videodev = getenv("VIDEO");  	if (getenv("AUDIO")) -		videodev = getenv("AUDIO"); +		audiodev = getenv("AUDIO");  	printf("using video device '%s'\n", videodev);  	printf("using audio device '%s'\n", audiodev); -	if ( (filefd = open(argv[1],O_RDONLY)) < 0){ +	putchar('\n'); + +	printf("Freeze       by pressing `z'\n"); +	printf("Stop         by pressing `s'\n"); +	printf("Continue     by pressing `c'\n"); +	printf("Start        by pressing `p'\n"); +	printf("FastForward  by pressing `f'\n"); +	printf("Mute         by pressing `m'\n"); +	printf("UnMute       by pressing `u'\n"); +	printf("MP2/AC3      by pressing `d'\n"); +	printf("SlowMotion   by pressing `l'\n"); +	printf("Quit         by pressing `q'\n"); + +	putchar('\n'); + +	errno = ENOENT; +	if (!argv[0] || (filefd = open(argv[0], O_RDONLY)) < 0) {  		perror("File open:");  		return -1;  	} -	if((vfd = open(videodev,O_RDWR|O_NONBLOCK)) < 0){ +	if ((vfd = open(videodev,O_RDWR|O_NONBLOCK)) < 0) {  		perror("VIDEO DEVICE: ");  		return -1;  	} -	if((afd = open(audiodev,O_RDWR|O_NONBLOCK)) < 0){ +	if ((afd = open(audiodev,O_RDWR|O_NONBLOCK)) < 0) {  		perror("AUDIO DEVICE: ");  		return -1;  	} +  	play_file_av(filefd, vfd, afd);  	close(vfd);  	close(afd);  	close(filefd);  	return 0; - -  } - diff --git a/test/test_dvr.c b/test/test_dvr.c index 4cee62f..f61f872 100644 --- a/test/test_dvr.c +++ b/test/test_dvr.c @@ -82,7 +82,7 @@ static void process_data(int dvrfd, int tsfd)  		fprintf(stderr, "got %d bytes (%llu total)\n", bytes, total_bytes);  } -static int add_filter(unsigned int pid, const unsigned char* dmxdev) +static int add_filter(unsigned int pid, const char* dmxdev)  {  	int fd;  	struct dmx_pes_filter_params f; @@ -161,4 +161,3 @@ int main(int argc, char *argv[])  	close(dvrfd);  	return 0;  } - diff --git a/test/test_dvr_play.c b/test/test_dvr_play.c index af963d9..7ab665f 100644 --- a/test/test_dvr_play.c +++ b/test/test_dvr_play.c @@ -141,4 +141,3 @@ int main(int argc, char **argv)  	close(filefd);  	return 0;  } - diff --git a/test/test_front.c b/test/test_front.c index 1d08fc6..72acbb6 100644 --- a/test/test_front.c +++ b/test/test_front.c @@ -1,4 +1,4 @@ -/*  +/*   * test_front.c - Test program for new API   *   * Copyright (C) 2000 Ralph  Metzler <ralph@convergence.de> @@ -40,7 +40,7 @@ int OSTSelftest(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,OST_SELFTEST,0) < 0)){ +	if ((ans = ioctl(fd,OST_SELFTEST,0)) < 0) {  		perror("OST SELF TEST: ");  		return -1;  	} @@ -52,7 +52,7 @@ int OSTSetPowerState(int fd, uint32_t state)  {  	int ans; -	if ( (ans = ioctl(fd,OST_SET_POWER_STATE,state) < 0)){ +	if ((ans = ioctl(fd,OST_SET_POWER_STATE,state)) < 0) {  		perror("OST SET POWER STATE: ");  		return -1;  	} @@ -64,7 +64,7 @@ int OSTGetPowerState(int fd, uint32_t *state)  {  	int ans; -	if ( (ans = ioctl(fd,OST_GET_POWER_STATE,state) < 0)){ +	if ((ans = ioctl(fd,OST_GET_POWER_STATE,state)) < 0) {  		perror("OST GET POWER STATE: ");  		return -1;  	} @@ -95,7 +95,7 @@ int FEReadStatus(int fd)  	int ans;  	feStatus stat; -	if ( (ans = ioctl(fd,FE_READ_STATUS,&stat) < 0)){ +	if ((ans = ioctl(fd,FE_READ_STATUS,&stat)) < 0) {  		perror("FE READ STATUS: ");  		return -1;  	} @@ -116,7 +116,7 @@ int FEReadBER(int fd, uint32_t *ber)  {  	int ans; -	if ( (ans = ioctl(fd,FE_READ_BER, ber) < 0)){ +	if ((ans = ioctl(fd,FE_READ_BER, ber)) < 0) {  		perror("FE READ_BER: ");  		return -1;  	} @@ -129,7 +129,7 @@ int FEReadSignalStrength(int fd, int32_t *strength)  {  	int ans; -	if ( (ans = ioctl(fd,FE_READ_SIGNAL_STRENGTH, strength) < 0)){ +	if ((ans = ioctl(fd,FE_READ_SIGNAL_STRENGTH, strength)) < 0) {  		perror("FE READ SIGNAL STRENGTH: ");  		return -1;  	} @@ -142,7 +142,7 @@ int FEReadSNR(int fd, int32_t *snr)  {  	int ans; -	if ( (ans = ioctl(fd,FE_READ_SNR, snr) < 0)){ +	if ((ans = ioctl(fd,FE_READ_SNR, snr)) < 0) {  		perror("FE READ_SNR: ");  		return -1;  	} @@ -156,7 +156,7 @@ int FEReadUncorrectedBlocks(int fd, uint32_t *ucb)  {  	int ans; -	if ( (ans = ioctl(fd,FE_READ_UNCORRECTED_BLOCKS, ucb) < 0)){ +	if ((ans = ioctl(fd,FE_READ_UNCORRECTED_BLOCKS, ucb)) < 0) {  		perror("FE READ UNCORRECTED BLOCKS: ");  		return -1;  	} @@ -169,7 +169,7 @@ int FEGetNextFrequency(int fd, uint32_t *nfr)  {  	int ans; -	if ( (ans = ioctl(fd,FE_GET_NEXT_FREQUENCY, nfr) < 0)){ +	if ((ans = ioctl(fd,FE_GET_NEXT_FREQUENCY, nfr)) < 0) {  		perror("FE GET NEXT FREQUENCY: ");  		return -1;  	} @@ -182,7 +182,7 @@ int FEGetNextSymbolRate(int fd, uint32_t *nsr)  {  	int ans; -	if ( (ans = ioctl(fd,FE_GET_NEXT_SYMBOL_RATE, nsr) < 0)){ +	if ((ans = ioctl(fd,FE_GET_NEXT_SYMBOL_RATE, nsr)) < 0) {  		perror("FE GET NEXT SYMBOL RATE: ");  		return -1;  	} @@ -195,7 +195,7 @@ int QPSKTune(int fd, struct qpskParameters *param)  {  	int ans; -	if ( (ans = ioctl(fd,QPSK_TUNE, param) < 0)){ +	if ((ans = ioctl(fd,QPSK_TUNE, param)) < 0) {  		perror("QPSK TUNE: ");  		return -1;  	} @@ -207,7 +207,7 @@ int QPSKGetEvent (int fd, struct qpskEvent *event)  {  	int ans; -	if ( (ans = ioctl(fd,QPSK_GET_EVENT, event) < 0)){ +	if ((ans = ioctl(fd,QPSK_GET_EVENT, event)) < 0) {  		perror("QPSK GET EVENT: ");  		return -1;  	} @@ -219,7 +219,7 @@ int QPSKFEInfo (int fd, struct qpskFrontendInfo *info)  {  	int ans; -	if ( (ans = ioctl(fd,QPSK_FE_INFO, info) < 0)){ +	if ((ans = ioctl(fd,QPSK_FE_INFO, info)) < 0) {  		perror("QPSK FE INFO: ");  		return -1;  	} @@ -238,7 +238,7 @@ int SecGetStatus (int fd, struct secStatus *state)  {  	int ans; -	if ( (ans = ioctl(fd,SEC_GET_STATUS, state) < 0)){ +	if ((ans = ioctl(fd,SEC_GET_STATUS, state)) < 0) {  		perror("QPSK GET EVENT: ");  		return -1;  	} @@ -261,7 +261,7 @@ int SecGetStatus (int fd, struct secStatus *state)  		break;  	} -	 +  	switch (state->selVolt){  	case SEC_VOLTAGE_OFF:  		printf("SEC VOLTAGE:  OFF (%d)\n",state->selVolt); @@ -299,11 +299,11 @@ main(int argc, char **argv)  	struct qpskFrontendInfo info;  	struct secStatus sec_state; -	if((fd = open("/dev/ost/qpskfe",O_RDWR)) < 0){ +	if ((fd = open("/dev/ost/qpskfe",O_RDWR)) < 0){  		perror("FRONTEND DEVICE: ");  		return -1;  	} -	if((fd_sec = open("/dev/ost/sec",O_RDWR)) < 0){ +	if ((fd_sec = open("/dev/ost/sec",O_RDWR)) < 0){  		perror("SEC DEVICE: ");  		return -1;  	} @@ -325,4 +325,3 @@ main(int argc, char **argv)  	close(fd);  	close(fd_sec);  } - diff --git a/test/test_sec_ne.c b/test/test_sec_ne.c index 9fcc8b3..350ac09 100644 --- a/test/test_sec_ne.c +++ b/test/test_sec_ne.c @@ -65,7 +65,7 @@ void process_section(int fd)  }  int set_filter(int fd, unsigned int pid, unsigned int tid, -		unsigned int tid_ext, unsigned int version_number) +		unsigned int tid_ext, unsigned int _version_number)  {  	struct dmx_sct_filter_params f;  	unsigned long bufsz; @@ -92,8 +92,8 @@ int set_filter(int fd, unsigned int pid, unsigned int tid,  		f.filter.mask[1]   = 0xff;  		f.filter.mask[2]   = 0xff;  	} -	if (version_number < 0x20) { -		f.filter.filter[3] = (uint8_t) (version_number << 1); +	if (_version_number < 0x20) { +		f.filter.filter[3] = (uint8_t) (_version_number << 1);  		f.filter.mask[3]   = 0x3e;  		f.filter.mode[3]   = 0x3e;  	} @@ -162,4 +162,3 @@ int main(int argc, char *argv[])  	close(dmxfd);  	return 0;  } - diff --git a/test/test_sections.c b/test/test_sections.c index d567db0..7f32cc8 100644 --- a/test/test_sections.c +++ b/test/test_sections.c @@ -178,7 +178,7 @@ int main(int argc, char *argv[])                  fprintf(stderr, "0x%.2x ", mask[filter_idx]);          fprintf(stderr, "\n"); -         +  	if ((dmxfd = open(dmxdev, O_RDWR)) < 0){  		perror("open");  		return 1; @@ -194,4 +194,3 @@ int main(int argc, char *argv[])  	close(dmxfd);  	return 0;  } - diff --git a/test/test_stc.c b/test/test_stc.c index 89f141d..7afa817 100644 --- a/test/test_stc.c +++ b/test/test_stc.c @@ -71,4 +71,3 @@ int main(int argc, char *argv[])  	close(dmxfd);  	return 0;  } - diff --git a/test/test_stillimage.c b/test/test_stillimage.c index 264b2c4..358c0f1 100644 --- a/test/test_stillimage.c +++ b/test/test_stillimage.c @@ -6,7 +6,7 @@   *   * or more advanced using netpbm and mpeg2enc (not mpeg2encode) :   *   $ cat image.jpg | jpegtopnm | pnmscale -xsize=704 -ysize=576 |\ - *      ppmntsc --pal | ppmtoy4m  -F 25:1 -A 4:3 |\ + *      ppmntsc --pal | ppmtoy4m  -F 25:1 -A 4:3 -S 420mpeg2 |\   *      mpeg2enc -f 7 -T 90 -F 3 -np -a 2 -o "image.mpg"   *   */ @@ -21,6 +21,7 @@  #include <time.h>  #include <unistd.h> +#include <linux/types.h>  #include <linux/dvb/video.h> @@ -82,7 +83,7 @@ next_pic:  		return -1;  	} -	printf ("read: %d bytes\n", read(filefd, sp.iFrame, sp.size)); +	printf ("read: %d bytes\n", (int) read(filefd, sp.iFrame, sp.size));  	close(filefd);  	if ((ioctl(fd, VIDEO_STILLPICTURE, &sp) < 0)) { @@ -99,5 +100,3 @@ next_pic:  	return 0;  } - - diff --git a/test/test_switch.c b/test/test_switch.c index 7f80891..b2939ad 100644 --- a/test/test_switch.c +++ b/test/test_switch.c @@ -1,4 +1,4 @@ -/*  +/*   * test_switch.c - Test program for new API   *   * Copyright (C) 2001 Ralph  Metzler <ralph@convergence.de> @@ -42,97 +42,97 @@ int fd_demuxv;  int fd_demuxtt;  struct dmx_sct_filter_params sctFilterParams; -	 +  struct secCommand scmd;  struct secCmdSequence scmds; -struct dmx_pes_filter_params pesFilterParams;  -struct dmx_pes_filter_params pesFilterParamsV;  -struct dmx_pes_filter_params pesFilterParamsA;  -struct dmx_pes_filter_params pesFilterParamsTT;  +struct dmx_pes_filter_params pesFilterParams; +struct dmx_pes_filter_params pesFilterParamsV; +struct dmx_pes_filter_params pesFilterParamsA; +struct dmx_pes_filter_params pesFilterParamsTT;  FrontendParameters frp;  int front_type; -set_front(void)  +set_front(void)  {  	fe_status_t stat = 0;  	int i, freq;  	uint32_t soff; -	   +  	scmds.miniCommand = SEC_MINI_NONE;  	scmds.numCommands = 1;  	scmds.commands = &scmd; -	 +  	soff = frp.u.qpsk.SymbolRate/16000;  	if (ioctl(fd_sec, SEC_SEND_SEQUENCE, &scmds) < 0)  		perror("setfront sec");  	usleep(100000); -		 +  	freq = frp.Frequency; -		 +  	while(tune_it(&frp)); -}	         +} -set_diseqc_nb(int nr)  +set_diseqc_nb(int nr)  {  	scmd.type=0;  	scmd.u.diseqc.addr = 0x10;  	scmd.u.diseqc.cmd = 0x38;  	scmd.u.diseqc.numParams = 1; -	scmd.u.diseqc.params[0] = 0xF0 | ((nr * 4) & 0x0F) |  +	scmd.u.diseqc.params[0] = 0xF0 | ((nr * 4) & 0x0F) |  	  (scmds.continuousTone == SEC_TONE_ON ? 1 : 0) |  	  (scmds.voltage==SEC_VOLTAGE_18 ? 2 : 0);  } -set_ttpid(ushort ttpid)  -{   +set_ttpid(ushort ttpid) +{          if (ttpid==0 || ttpid==0xffff || ttpid==0x1fff) {  	        ioctl(fd_demuxtt, DMX_STOP, 0);  	        return;  	}  	pesFilterParamsTT.pid     = ttpid; -	pesFilterParamsTT.input   = DMX_IN_FRONTEND;  -	pesFilterParamsTT.output  = DMX_OUT_DECODER;  -	pesFilterParamsTT.pes_type = DMX_PES_TELETEXT;  +	pesFilterParamsTT.input   = DMX_IN_FRONTEND; +	pesFilterParamsTT.output  = DMX_OUT_DECODER; +	pesFilterParamsTT.pes_type = DMX_PES_TELETEXT;  	pesFilterParamsTT.flags   = DMX_IMMEDIATE_START; -	if (ioctl(fd_demuxtt, DMX_SET_PES_FILTER,  +	if (ioctl(fd_demuxtt, DMX_SET_PES_FILTER,  		  &pesFilterParamsTT) < 0) {  	  printf("PID=%04x\n", ttpid);  		perror("set_ttpid");  	}  } -set_vpid(ushort vpid)  -{   +set_vpid(ushort vpid) +{          if (vpid==0 || vpid==0xffff || vpid==0x1fff) {  	        ioctl(fd_demuxv, DMX_STOP, 0);  	        return;  	}  	pesFilterParamsV.pid     = vpid; -	pesFilterParamsV.input   = DMX_IN_FRONTEND;  -	pesFilterParamsV.output  = DMX_OUT_DECODER;  -	pesFilterParamsV.pes_type = DMX_PES_VIDEO;  +	pesFilterParamsV.input   = DMX_IN_FRONTEND; +	pesFilterParamsV.output  = DMX_OUT_DECODER; +	pesFilterParamsV.pes_type = DMX_PES_VIDEO;  	pesFilterParamsV.flags   = DMX_IMMEDIATE_START; -	if (ioctl(fd_demuxv, DMX_SET_PES_FILTER,  +	if (ioctl(fd_demuxv, DMX_SET_PES_FILTER,  		  &pesFilterParamsV) < 0)  		perror("set_vpid");  } -set_apid(ushort apid)  -{   +set_apid(ushort apid) +{          if (apid==0 || apid==0xffff || apid==0x1fff) {  	        ioctl(fd_demuxa, DMX_STOP, apid);  	        return;  	}  	pesFilterParamsA.pid = apid; -	pesFilterParamsA.input = DMX_IN_FRONTEND;  -	pesFilterParamsA.output = DMX_OUT_DECODER;  -	pesFilterParamsA.pes_type = DMX_PES_AUDIO;  +	pesFilterParamsA.input = DMX_IN_FRONTEND; +	pesFilterParamsA.output = DMX_OUT_DECODER; +	pesFilterParamsA.pes_type = DMX_PES_AUDIO;  	pesFilterParamsA.flags = DMX_IMMEDIATE_START; -	if (ioctl(fd_demuxa, DMX_SET_PES_FILTER,  +	if (ioctl(fd_demuxa, DMX_SET_PES_FILTER,  		  &pesFilterParamsA) < 0)  		perror("set_apid");  } @@ -156,7 +156,7 @@ int tune_it(FrontendParameters *frp)  	if (poll(pfd,1,3000)){  		if (pfd[0].revents & POLLIN){  			printf("Getting QPSK event\n"); -			if ( ioctl(fd_frontend, FE_GET_EVENT, &event)   +			if ( ioctl(fd_frontend, FE_GET_EVENT, &event)  			     == -EOVERFLOW){  				perror("qpsk get event"); @@ -170,7 +170,7 @@ int tune_it(FrontendParameters *frp)  			case FE_FAILURE_EV:  				printf("failure event\n");  				return -1; -				 +  			case FE_COMPLETION_EV:  				printf("completion event\n");  			} @@ -179,7 +179,7 @@ int tune_it(FrontendParameters *frp)  	return 0;  } -set_tp(uint *freq, int ttk, int pol, uint srate, int dis)  +set_tp(uint *freq, int ttk, int pol, uint srate, int dis)  {  	if (*freq < 11700000) {  		frp.Frequency = (*freq - 9750000); @@ -195,22 +195,22 @@ set_tp(uint *freq, int ttk, int pol, uint srate, int dis)  	frp.u.qpsk.FEC_inner = 0;  } -get_front(void)  +get_front(void)  {  	set_vpid(0);  	set_apid(0);  	set_ttpid(0); -	scmds.voltage = SEC_VOLTAGE_18;  +	scmds.voltage = SEC_VOLTAGE_18;  	scmds.miniCommand = SEC_MINI_NONE;  	scmds.continuousTone = SEC_TONE_OFF;  	scmds.numCommands=1;  	scmds.commands=&scmd; -	 +  	frp.Frequency=(12073000-10600000);  	frp.u.qpsk.SymbolRate=25378000;  	frp.u.qpsk.FEC_inner=(fe_code_rate_t)5; -}	         -   +} +  void get_sect(int fd)  { @@ -219,10 +219,10 @@ void get_sect(int fd)          uint16_t cpid = 0;          uint16_t length;          struct pollfd pfd; -        +          pfd.fd = fd;          pfd.events = POLLIN; -        +  	while (1){  		if (poll(&pfd, 1, 5000) != POLLIN) {  			printf("not found\n"); @@ -232,7 +232,7 @@ void get_sect(int fd)  			length  = (sec[1]& 0x0F)<<8;  			length |= (sec[2]& 0xFF); -         +  			for (i= 0; i < length+3; i++) {  				printf("0x%02x ",sec[i]); @@ -248,7 +248,7 @@ int FEReadStatus(int fd, fe_status_t *stat)  {  	int ans; -	if ( (ans = ioctl(fd,FE_READ_STATUS,stat) < 0)){ +	if ((ans = ioctl(fd,FE_READ_STATUS,stat)) < 0) {  		perror("FE READ STATUS: ");  		return -1;  	} @@ -268,7 +268,7 @@ int FEReadStatus(int fd, fe_status_t *stat)  int has_signal()  {  	fe_status_t stat; -	 +  	FEReadStatus(fd_frontend, &stat);  	if (stat & FE_HAS_SIGNAL)  		return 1; @@ -304,7 +304,7 @@ main()  	scmds.commands=&scmd;  	frp.Frequency = (12073000 - 10600000);  //	frp.Frequency = (11975000 - 10600000); -	scmds.continuousTone = SEC_TONE_ON;  +	scmds.continuousTone = SEC_TONE_ON;  	frp.u.qpsk.SymbolRate = 25378000;  //	frp.u.qpsk.SymbolRate = 27500000;  //	frp.u.qpsk.FEC_inner = FEC_AUTO; @@ -313,7 +313,7 @@ main()  	scmd.u.diseqc.addr = 0x10;  	scmd.u.diseqc.cmd = 0x38;  	scmd.u.diseqc.numParams = 1; -	scmd.u.diseqc.params[0] = 0xF0 | ((3 * 4) & 0x0F) |  +	scmd.u.diseqc.params[0] = 0xF0 | ((3 * 4) & 0x0F) |  	  (scmds.continuousTone == SEC_TONE_ON ? 1 : 0) |  	  (scmds.voltage==SEC_VOLTAGE_18 ? 2 : 0); @@ -324,7 +324,7 @@ main()  /* -	if ((fd_demuxa=open("/dev/ost/demux", O_RDWR|O_NONBLOCK))  +	if ((fd_demuxa=open("/dev/ost/demux", O_RDWR|O_NONBLOCK))  	    < 0){  		perror("DEMUX DEVICE: ");  		return -1; @@ -336,18 +336,18 @@ main()          sctFilterParams.timeout                   = 0;          sctFilterParams.flags                     = DMX_IMMEDIATE_START; -        if (ioctl(fd_demuxa, DMX_SET_FILTER, &sctFilterParams) < 0)   +        if (ioctl(fd_demuxa, DMX_SET_FILTER, &sctFilterParams) < 0)                  perror("DMX SET FILTER:");  	get_sect(fd_demuxa);  */  	pesFilterParamsA.pid = 0x1fff; -	pesFilterParamsA.input = DMX_IN_FRONTEND;  -	pesFilterParamsA.output = DMX_OUT_TS_TAP;  -	pesFilterParamsA.pes_type = DMX_PES_OTHER;  +	pesFilterParamsA.input = DMX_IN_FRONTEND; +	pesFilterParamsA.output = DMX_OUT_TS_TAP; +	pesFilterParamsA.pes_type = DMX_PES_OTHER;  	pesFilterParamsA.flags = DMX_IMMEDIATE_START; -	if (ioctl(fd_demuxa, DMX_SET_PES_FILTER,  +	if (ioctl(fd_demuxa, DMX_SET_PES_FILTER,  		  &pesFilterParamsA) < 0)  		perror("set_apid"); diff --git a/test/test_tt.c b/test/test_tt.c index 6511f39..da288d3 100644 --- a/test/test_tt.c +++ b/test/test_tt.c @@ -202,4 +202,3 @@ int main(int argc, char *argv[])  	close(dmxfd);  	return 0;  } - diff --git a/test/test_vevent.c b/test/test_vevent.c index f61143e..bfd0369 100644 --- a/test/test_vevent.c +++ b/test/test_vevent.c @@ -1,4 +1,4 @@ -/*  +/*   * test_vevent.c - Test VIDEO_GET_EVENT and poll(9 for video events   *   * Copyright (C) 2003 convergence GmbH @@ -29,6 +29,7 @@  #include <time.h>  #include <unistd.h> +#include <linux/types.h>  #include <linux/dvb/video.h> @@ -73,7 +74,7 @@ int main(void)  	printf("using video device '%s'\n", videodev); -	if((vfd = open(videodev, O_RDONLY | O_NONBLOCK)) < 0) { +	if ((vfd = open(videodev, O_RDONLY | O_NONBLOCK)) < 0) {  		perror("open video device");  		return 1;  	} @@ -122,4 +123,3 @@ int main(void)  	close(vfd);  	return 0;  } - diff --git a/test/test_video.c b/test/test_video.c index 09c2be1..ac66050 100644 --- a/test/test_video.c +++ b/test/test_video.c @@ -1,4 +1,4 @@ -/*  +/*   * test_video.c - Test program for new API   *   * Copyright (C) 2000 Ralph  Metzler <ralph@convergence.de> @@ -40,7 +40,7 @@ int videoStop(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_STOP,0) < 0)){ +	if ((ans = ioctl(fd,VIDEO_STOP,0)) < 0) {  		perror("VIDEO STOP: ");  		return -1;  	} @@ -52,7 +52,7 @@ int videoPlay(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_PLAY) < 0)){ +	if ((ans = ioctl(fd,VIDEO_PLAY)) < 0) {  		perror("VIDEO PLAY: ");  		return -1;  	} @@ -65,7 +65,7 @@ int videoFreeze(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_FREEZE) < 0)){ +	if ((ans = ioctl(fd,VIDEO_FREEZE)) < 0) {  		perror("VIDEO FREEZE: ");  		return -1;  	} @@ -78,7 +78,7 @@ int videoContinue(int fd)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_CONTINUE) < 0)){ +	if ((ans = ioctl(fd,VIDEO_CONTINUE)) < 0) {  		perror("VIDEO CONTINUE: ");  		return -1;  	} @@ -90,7 +90,7 @@ int videoSelectSource(int fd, video_stream_source_t source)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_SELECT_SOURCE, source) < 0)){ +	if ((ans = ioctl(fd,VIDEO_SELECT_SOURCE, source)) < 0) {  		perror("VIDEO SELECT SOURCE: ");  		return -1;  	} @@ -104,7 +104,7 @@ int videoSetBlank(int fd, boolean state)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_SET_BLANK, state) < 0)){ +	if ((ans = ioctl(fd,VIDEO_SET_BLANK, state)) < 0) {  		perror("VIDEO SET BLANK: ");  		return -1;  	} @@ -116,7 +116,7 @@ int videoFastForward(int fd,int nframes)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_FAST_FORWARD, nframes) < 0)){ +	if ((ans = ioctl(fd,VIDEO_FAST_FORWARD, nframes)) < 0) {  		perror("VIDEO FAST FORWARD: ");  		return -1;  	} @@ -128,7 +128,7 @@ int videoSlowMotion(int fd,int nframes)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_SLOWMOTION, nframes) < 0)){ +	if ((ans = ioctl(fd,VIDEO_SLOWMOTION, nframes)) < 0) {  		perror("VIDEO SLOWMOTION: ");  		return -1;  	} @@ -141,7 +141,7 @@ int videoGetStatus(int fd)  	struct video_status stat;  	int ans; -	if ( (ans = ioctl(fd,VIDEO_GET_STATUS, &stat) < 0)){ +	if ((ans = ioctl(fd,VIDEO_GET_STATUS, &stat)) < 0) {  		perror("VIDEO GET STATUS: ");  		return -1;  	} @@ -164,7 +164,7 @@ int videoGetStatus(int fd)  		printf("unknown (%d)\n",stat.play_state);  		break;  	} -	 +  	printf("  Stream Source        : ");  	switch((int)stat.stream_source){  	case VIDEO_SOURCE_DEMUX: @@ -213,7 +213,7 @@ int videoStillPicture(int fd, struct video_still_picture *sp)  {  	int ans; -	if ( (ans = ioctl(fd,VIDEO_STILLPICTURE, sp) < 0)){ +	if ((ans = ioctl(fd,VIDEO_STILLPICTURE, sp)) < 0) {  		perror("VIDEO STILLPICTURE: ");  		return -1;  	} @@ -234,17 +234,17 @@ void play_file_video(int filefd, int fd)  	pfd[0].fd = STDIN_FILENO;  	pfd[0].events = POLLIN; -	 +  	pfd[1].fd = fd;  	pfd[1].events = POLLOUT; -	 +  	videoSelectSource(fd,VIDEO_SOURCE_MEMORY);  	videoPlay(fd); -	 -	 + +  	count = read(filefd,buf,BUFFY);  	write(fd,buf,count); -	 +  	while ( (count = read(filefd,buf,BUFFY)) >= 0  ){  		written = 0;  		while(written < count){ @@ -267,7 +267,7 @@ void play_file_video(int filefd, int fd)  						printf("playback stopped\n");  						stopped = 1;  						break; -						 +  					case 'c':  						videoContinue(fd);  						printf("playback continued\n"); @@ -298,7 +298,7 @@ void play_file_video(int filefd, int fd)  						break;  					}  				} -				 +  			}  		}  	} @@ -310,12 +310,12 @@ void load_iframe(int filefd, int fd)  	struct video_still_picture sp;  	fstat(filefd, &st); -	 +  	sp.iFrame = (char *) malloc(st.st_size);  	sp.size = st.st_size;  	printf("I-frame size: %d\n", sp.size); -	 -	if(!sp.iFrame) { + +	if (!sp.iFrame) {  		printf("No memory for I-Frame\n");  		return;  	} @@ -338,12 +338,12 @@ main(int argc, char **argv)  		perror("File open:");  		return -1;  	} -	if((fd = open("/dev/ost/video1",O_RDWR|O_NONBLOCK)) < 0){ +	if ((fd = open("/dev/ost/video1",O_RDWR|O_NONBLOCK)) < 0){  		perror("VIDEO DEVICE: ");  		return -1;  	} -	     -	 + +  //	videoSetBlank(fd,false); @@ -365,4 +365,3 @@ main(int argc, char **argv)  } - diff --git a/test/video.c b/test/video.c index ea9e4a2..fae421e 100644 --- a/test/video.c +++ b/test/video.c @@ -1,13 +1,14 @@ - /** - *  A tiny video watching application, just starts capturing /dev/video - *  into /dev/fb0. - *  Be shure to have >8Bit/pixel color resolution and r/w access for  - *  /dev/video0, /dev/fb0 and /dev/tty0 to let this work... - * - *   compile with - * - *   $ gcc -g -Wall -O2 -o video video.c -I../../ost/include - */ +#define USAGE \ +"\n" \ +"\n A tiny video watching application, just starts capturing /dev/video" \ +"\n into /dev/fb0." \ +"\n Be shure to have >8Bit/pixel color resolution and r/w access for " \ +"\n /dev/video0, /dev/fb0 and /dev/tty0 to let this work..." \ +"\n" \ +"\n  compile with" \ +"\n" \ +"\n  $ gcc -g -Wall -O2 -o video video.c -I../../ost/include" \ +"\n"  #include <sys/mman.h>  #include <sys/ioctl.h> @@ -55,7 +56,7 @@ int init_fb (void)  	}  	write( vt_fd, blankoff_str, strlen(blankoff_str) ); -	 +  	if (ioctl (fd, FBIOGET_VSCREENINFO, &fb_var) < 0) {  		perror("Could not get variable screen information (fb_var)\n");  		return 1; @@ -77,14 +78,14 @@ int init_video (int stop)  	struct video_capability vcap;  	if ((fd = open (video_devname, O_RDWR)) < 0) { -		fprintf (stderr,  +		fprintf (stderr,  			 "%s: Could not open %s, please check permissions\n",  			 __FUNCTION__, video_devname);  		return -1;  	}  	ioctl(fd, VIDIOCGCAP, &vcap); -	 +  	if (ioctl(fd, VIDIOCCAPTURE, &zero) < 0) {  		perror("Could not stop capturing (VIDIOCCAPTURE failed)\n");  		return -2; @@ -92,7 +93,7 @@ int init_video (int stop)  	if (stop)  		return 0; -	 +  	{  		struct video_buffer b;  		b.base = (void*) fb_fix.smem_start; @@ -105,7 +106,7 @@ int init_video (int stop)  			return -3;  		}  	} -	 +  	{  		struct video_picture p;  		if (ioctl(fd, VIDIOCGPICT, &p) < 0) { @@ -131,7 +132,7 @@ int init_video (int stop)  			return -5;  		}  	} -	 +  	{  		struct video_window win;  		win.width = min((__u32) vcap.maxwidth, fb_var.xres); @@ -147,14 +148,14 @@ int init_video (int stop)  			return -6;  		}  	} -	 +  	if (ioctl(fd, VIDIOCCAPTURE, &one) < 0) {  		perror("Could not start capturing (VIDIOCCAPTURE failed)\n");  		return -7;  	} -	 +  	close (fd); -	 +  	return 0;  } @@ -173,10 +174,9 @@ int main (int argc, char **argv)  		video_devname = argv[1];  	if (argc != 1 && argc != 2 && !(argc == 3 && stop)) { -		fprintf (stderr, "usage: %s <devname> <stop>\n", argv[0]); +		fprintf(stderr, "usage: %s <devname> <stop>\n" USAGE, argv[0]);  		exit (-1);  	}  	return init_video (stop);  } - | 
