aboutsummaryrefslogtreecommitdiffhomepage
path: root/stream
diff options
context:
space:
mode:
Diffstat (limited to 'stream')
-rw-r--r--stream/tv.h6
-rw-r--r--stream/tvi_vbi.c65
2 files changed, 70 insertions, 1 deletions
diff --git a/stream/tv.h b/stream/tv.h
index 202abd0012..98f9417f2e 100644
--- a/stream/tv.h
+++ b/stream/tv.h
@@ -280,6 +280,11 @@ typedef struct tt_char_s{
unsigned char raw; ///< raw character (as received from device)
} tt_char;
+typedef struct tt_link_s{
+ int pagenum; ///< page number
+ int subpagenum; ///< subpage number
+} tt_link_t;
+
typedef struct tt_page_s{
int pagenum; ///< page number
int subpagenum; ///< subpage number
@@ -288,6 +293,7 @@ typedef struct tt_page_s{
unsigned char flags; ///< page flags, not used
unsigned char raw[VBI_ROWS*VBI_COLUMNS]; ///< page data
struct tt_page_s* next_subpage;
+ struct tt_link_s links[6];
} tt_page;
typedef struct tt_stream_props_s{
diff --git a/stream/tvi_vbi.c b/stream/tvi_vbi.c
index d709ae567f..2c9213e504 100644
--- a/stream/tvi_vbi.c
+++ b/stream/tvi_vbi.c
@@ -376,7 +376,7 @@ static int steppage(int p, int direction, int skip_hidden)
*/
static void put_to_cache(priv_vbi_t* priv,tt_page* pg,int line){
tt_page* pgc; //page in cache
- int i,count;
+ int i,j,count;
if(line<0){
i=0;
@@ -406,6 +406,8 @@ static void put_to_cache(priv_vbi_t* priv,tt_page* pg,int line){
pgc->subpagenum=pg->subpagenum;
pgc->lang=pg->lang;
pgc->flags=pg->flags;
+ for(j=0;j<6;++j)
+ pgc->links[j]=pg->links[j];
//instead of copying entire page into cache, copy only undamaged
//symbols into cache
for(;i<count;i++){
@@ -1015,6 +1017,42 @@ static void decode_pkt_page(priv_vbi_t* priv,unsigned char*data,int magAddr,int
}
/**
+ * \brief decode packets 27 (teletext links)
+ * \param priv private data structure
+ * \param data raw teletext data
+ * \param magAddr teletext page's magazine address
+ */
+static int decode_pkt27(priv_vbi_t* priv,unsigned char* data,int magAddr){
+ int i,hpg;
+
+ if (!priv->mag[magAddr].pt)
+ return 0;
+ for(i=0;i<38;++i)
+ if ((data[i] = corrHamm48[ data[i] ]) & 0x80){
+ pll_add(priv,2,4);
+ return 0;
+ }
+
+ /*
+ Not a X/27/0 Format 1 packet or
+ flag "show links on row 24" is not set.
+ */
+ if (data[0] || !(data[37] & 8))
+ return 1;
+ for(i=0;i<6;++i) {
+ hpg = (magAddr<<8) ^ ((data[4+i*6]&0x8)<<5 | (data[6+i*6]&0xc)<<7);
+ if (!hpg) hpg=0x800;
+ priv->mag[magAddr].pt->links[i].pagenum = (data[1+i*6] & 0xf) |
+ ((data[2+i*6] & 0xf) << 4) | hpg;
+ priv->mag[magAddr].pt->links[i].subpagenum = ((data[3+i*6] & 0xf) |
+ (data[4+i*6] & 0xf) << 4 | (data[5+i*6] & 0xf) << 8 |
+ (data[6+i*6] & 0xf) << 12) & 0x3f7f;
+ }
+ put_to_cache(priv,priv->mag[magAddr].pt,-1);
+ return 1;
+}
+
+/**
* \brief decodes raw vbi data (signal amplitudes) into sequence of bytes
* \param priv private data structure
* \param buf raw vbi data (one line of frame)
@@ -1205,6 +1243,8 @@ static void vbi_decode(priv_vbi_t* priv,unsigned char*buf){
}else if(pkt>0 && pkt<VBI_ROWS){
if(!priv->mag[magAddr].pt) continue;
decode_pkt_page(priv,data+2,magAddr,pkt);//skip MRGA
+ }else if(pkt==27) {
+ decode_pkt27(priv,data+2,magAddr);
}else if(pkt==30){
decode_pkt30(priv,data+2,magAddr);
} else {
@@ -1339,6 +1379,7 @@ int teletext_control(void* p, int cmd, void *arg)
{
int fine_tune=99;
priv_vbi_t* priv=(priv_vbi_t*)p;
+ tt_page* pgc;
if (!priv && cmd!=TV_VBI_CONTROL_START)
return TVI_CONTROL_FALSE;
@@ -1424,6 +1465,28 @@ int teletext_control(void* p, int cmd, void *arg)
pthread_mutex_unlock(&(priv->buffer_mutex));
return TVI_CONTROL_TRUE;
}
+ case TV_VBI_CONTROL_GO_LINK:
+ {
+ int val=*(int *) arg;
+ if(val<1 || val>6)
+ return TVI_CONTROL_FALSE;
+ pthread_mutex_lock(&(priv->buffer_mutex));
+ if (!(pgc = priv->ptt_cache[priv->pagenum])) {
+ pthread_mutex_unlock(&(priv->buffer_mutex));
+ return TVI_CONTROL_FALSE;
+ }
+ if (!pgc->links[val-1].pagenum || pgc->links[val-1].pagenum>0x7ff) {
+ pthread_mutex_unlock(&(priv->buffer_mutex));
+ return TVI_CONTROL_FALSE;
+ }
+ priv->pagenum=pgc->links[val-1].pagenum;
+ if(pgc->links[val-1].subpagenum!=0x3f7f)
+ priv->subpagenum=pgc->links[val-1].subpagenum;
+ else
+ priv->subpagenum=get_subpagenum_from_cache(priv,priv->pagenum);
+ pthread_mutex_unlock(&(priv->buffer_mutex));
+ return TVI_CONTROL_TRUE;
+ }
case TV_VBI_CONTROL_SET_PAGE:
{
int val=*(int *) arg;