OSDN Git Service

stop using trunk directory in rectool
[rec10/rec10-git.git] / epgdump / sdt.c
index 46a74ff..6192f6b 100755 (executable)
@@ -1,3 +1,5 @@
+// -*- tab-width:4 -*-
+
 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
@@ -46,10 +48,10 @@ int parseSDTbody(unsigned char *data, SDTbody *b) {
 }
 
 int parseSVCdesc(unsigned char *data, SVCdesc *desc) {
+//0x48のサービス記述子、放送局の名前などが入っているよう
        int boff = 0;
-  
-       memset(desc, 0, sizeof(SVCdesc));
 
+       memset(desc, 0, sizeof(SVCdesc));
        desc->descriptor_tag = getBit(data, &boff, 8);
        desc->descriptor_length = getBit(data, &boff, 8);
        desc->service_type = getBit(data, &boff, 8);
@@ -60,7 +62,36 @@ int parseSVCdesc(unsigned char *data, SVCdesc *desc) {
 
        return desc->descriptor_length + 2;
 }
-int            serachid(SVT_CONTROL *top, int service_id)
+
+int parseLOGdesc(unsigned char *data, LOGdesc *desc) {
+//0xC1のデジタルコピー制御記述子
+//0xCFのロゴ伝送記述子
+       int boff = 0;
+
+       memset(desc, 0, sizeof(LOGdesc));
+       desc->descriptor_tag = getBit(data, &boff, 8);
+       desc->descriptor_length = getBit(data, &boff, 8);
+       desc->logo_transmission_type = getBit(data, &boff, 8);
+       if ( desc->logo_transmission_type == 0x01 ) {
+               desc->reserved_future_use1 = getBit(data, &boff, 7);
+               desc->logo_id = getBit(data, &boff, 9);
+               desc->reserved_future_use2 = getBit(data, &boff, 4);
+               desc->logo_version = getBit(data, &boff, 12);
+               desc->download_data_id = getBit(data, &boff, 16);
+       }
+       else if ( desc->logo_transmission_type == 0x02 ) {
+               desc->reserved_future_use1 = getBit(data, &boff, 7);
+               desc->logo_id = getBit(data, &boff, 9);
+       }
+       else if ( desc->logo_transmission_type == 0x03 ) {
+               memcpy(desc->logo_char, data + boff / 8, desc->descriptor_length);
+               // getStr(desc->logo_char, data, &boff, desc->descriptor_length);
+       }
+
+       return desc->descriptor_length + 2;
+}
+
+int serachid(SVT_CONTROL *top, int service_id)
 {
        SVT_CONTROL     *cur = top ;
        while(cur != NULL){
@@ -72,7 +103,7 @@ int          serachid(SVT_CONTROL *top, int service_id)
        return 0 ;
 }
 
-void   enqueue_sdt(SVT_CONTROL *top, SVT_CONTROL *sdtptr)
+void enqueue_sdt(SVT_CONTROL *top, SVT_CONTROL *sdtptr)
 {
        SVT_CONTROL     *cur ;
 
@@ -103,64 +134,148 @@ void     enqueue_sdt(SVT_CONTROL *top, SVT_CONTROL *sdtptr)
 
 }
 
-void dumpSDT(unsigned char *ptr, SVT_CONTROL *top)
+void dumpSDT(unsigned char *ptr, SVT_CONTROL *top,STATION **station, int * station_count,char *ontvheader)
 {
 
        SDThead  sdth;
        SDTbody  sdtb;
        SVCdesc  desc;
+       LOGdesc  logd;
        SVT_CONTROL     *svtptr ;
+       STATION * pStation = *station;
        int             rc ;
 
        int len = 0;
        int loop_len = 0;
+       int desc_len = 0;
+       char sid[80];
+       int stationi = *station_count;
+       int i = 0;
 
        /* SDT */
        len = parseSDThead(ptr, &sdth); 
        ptr += len;
-       loop_len = sdth.section_length - (len - 3 + 4); // 3¤Ï¶¦Ḁ̈إåÀĹ 4¤ÏCRC
+       loop_len = sdth.section_length - (len - 3 + 4); // 3は共通ヘッダ長 4はCRC
        while(loop_len > 0) {
                len = parseSDTbody(ptr, &sdtb);
+               //printf("body %d - %d = %d\n",loop_len,len,loop_len - len);
                ptr += len;
                loop_len -= len;
-               parseSVCdesc(ptr, &desc);
-
-               rc = serachid(top, sdtb.service_id);
-               if(rc == 0){
-                       svtptr = calloc(1, sizeof(SVT_CONTROL));
-                       svtptr->event_id = sdtb.service_id;
-                       svtptr->original_network_id = sdth.original_network_id;
-                       svtptr->transport_stream_id = sdth.transport_stream_id;
-                       svtptr->event_id = sdtb.service_id;
-                       memcpy(svtptr->servicename, desc.service_name, strlen(desc.service_name));
-                       enqueue_sdt(top, svtptr);
+
+               desc_len = sdtb.descriptors_loop_length;
+               loop_len -= desc_len;
+               while(desc_len > 0) {
+                       if ( *ptr == 0xCF ) {
+                               len = parseLOGdesc(ptr, &logd);
+                               ptr += len;
+                               desc_len -= len;
+
+                               #if 0
+                               printf("LOG=%d,%d,%d,%d\n",
+                                       logd.logo_transmission_type, logd.logo_id, 
+                                       logd.logo_version, logd.download_data_id);
+                               #endif
+
+                               /*
+                                       logd.logo_transmission_type
+                                       0x01 CDT 伝送方式1:CDT をダウンロードデータ識別で直接参照する場合
+                                       0x02 CDT 伝送方式2:CDT をロゴ識別を用いてダウンロードデータ識別を間接的に参照する場合
+                                       0x03 簡易ロゴ方式
+                               */
+                               if ( logd.logo_transmission_type != 0x01 ) continue;
+                               for (i=0; i<stationi; i++) {
+                                       // FIXME: pStation[i].logo_version < logd.logo_version
+                                       if ( pStation[i].svId == sdtb.service_id ) {
+                                               pStation[i].logo_download_data_id = logd.download_data_id;
+                                               pStation[i].logo_version = logd.logo_version;
+                                       }
+                               }
+
+                               continue;
+                       }
+                       else if ( *ptr != 0x48 ) {
+                               len = parseOTHERdesc(ptr);
+                               ptr += len;
+                               desc_len -= len;
+                               continue;
+                       }
+                       len = parseSVCdesc(ptr, &desc);
+                       //printf("desc %d - %d = %d\n",desc_len,len,desc_len - len);
+                       ptr += len;
+                       desc_len -= len;
+
+                       rc = serachid(top, sdtb.service_id);
+                       if(rc == 0){
+                               svtptr = calloc(1, sizeof(SVT_CONTROL));
+                               svtptr->original_network_id = sdth.original_network_id;
+                               svtptr->transport_stream_id = sdth.transport_stream_id;
+                               svtptr->event_id = sdtb.service_id;
+                               memcpy(svtptr->servicename, desc.service_name, strlen(desc.service_name));
+                               if (desc.service_type == 1){
+                                       enqueue_sdt(top, svtptr);
+
+                                       sprintf(sid, "%s_%d", ontvheader, sdtb.service_id );
+                                       for (i=0; i<stationi; i++) {
+                                               if ( !strcmp( pStation[i].ontv, sid ) ) {
+                                                       // 既に同じものが含まれるのでスキップ
+                                                       break;
+                                               }
+                                       }
+
+                                       pStation = realloc(pStation, (stationi + 1) * sizeof(STATION));
+                                       memset(&pStation[stationi], 0, sizeof(STATION));
+
+                                       pStation[stationi].name = malloc( strlen(desc.service_name) + 1 );
+                                       pStation[stationi].ontv = malloc( strlen(sid) + 1 );
+                                       pStation[stationi].tsId = sdth.transport_stream_id;
+                                       pStation[stationi].onId = sdth.original_network_id;
+                                       pStation[stationi].svId = sdtb.service_id;
+
+                                       strcpy(pStation[stationi].name, desc.service_name);
+                                       strcpy(pStation[stationi].ontv, sid);
+
+                                       stationi++;
+                               }
+
 #if 0
-                       printf("SDT=%s,%d,%x,%x,%x,%x,%x,%x,%x\n",
-                               desc.service_name, sdtb.service_id, sdtb.reserved_future_use1,
-                               sdtb.EIT_user_defined_flags, sdtb.EIT_schedule_flag, sdtb.EIT_present_following_flag,
-                               sdtb.running_status, sdtb.free_CA_mode, sdtb.descriptors_loop_length);
+                               printf("STATION=%s,%d,%d,%d,%d,%d\n",
+                                       desc.service_name,sdtb.service_id,sdth.transport_stream_id,
+                                       sdth.original_network_id,sdtb.service_id,desc.service_type);
+
+                               printf("SDT=%s,%d,%x,%x,%x,%x,%x,%x,%x\n",
+                                       desc.service_name, sdtb.service_id, sdtb.reserved_future_use1,
+                                       sdtb.EIT_user_defined_flags, sdtb.EIT_schedule_flag, sdtb.EIT_present_following_flag,
+                                       sdtb.running_status, sdtb.free_CA_mode, sdtb.descriptors_loop_length);
 /*
-#else
-0x01:\83f\83W\83^\83\8bTV\83T\81[\83r\83X
-0xA5:\83v\83\8d\83\82\81[\83V\83\87\83\93\89f\91\9c\83T\81[\83r\83X
-0x0C:\83f\81[\83^\83T\81[\83r\83X
- */
-                       printf("SDT=(%x:%x)%s,%d,%d,%d,%d,%d(%d,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x)\n",
-                                       sdth.table_id, desc.service_type, 
-                                       desc.service_name, sdtb.service_id,
-                                       desc.descriptor_tag, desc.descriptor_length, desc.service_type,
-                                       desc.service_provider_name_length, desc.service_name_length,
-                                       sdth.table_id, sdth.section_syntax_indicator, sdth.reserved_future_use1,
-                                       sdth.reserved1, sdth.section_length, sdth.transport_stream_id,
-                                       sdth.reserved2, sdth.version_number, sdth.current_next_indicator,
-                                       sdth.section_number, sdth.last_section_number, sdth.original_network_id,
-                                       sdth.reserved_future_use2);
+0x01:デジタルTVサービス
+0xA5:プロモーション映像サービス
+0x0C:データサービス
+*/
+                               printf("SDT=(%x:%x)%s,%d,%d,%d,%d,%d(%d,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x,%x)\n",
+                                               sdth.table_id, desc.service_type, 
+                                               desc.service_name, sdtb.service_id,
+                                               desc.descriptor_tag, desc.descriptor_length, desc.service_type,
+                                               desc.service_provider_name_length, desc.service_name_length,
+                                               sdth.table_id, sdth.section_syntax_indicator, sdth.reserved_future_use1,
+                                               sdth.reserved1, sdth.section_length, sdth.transport_stream_id,
+                                               sdth.reserved2, sdth.version_number, sdth.current_next_indicator,
+                                               sdth.section_number, sdth.last_section_number, sdth.original_network_id,
+                                               sdth.reserved_future_use2);
 #endif
+                       }
                }
-
-               ptr += sdtb.descriptors_loop_length;
+/*
+               //ptr += sdtb.descriptors_loop_length;
                loop_len -= sdtb.descriptors_loop_length;
+               
+               if (loop_len>0){
+                       ptr += sdtb.descriptors_loop_length;
+               }
+*/
        }
-  
+       *station = pStation;
+       *station_count = stationi;
+       //printf("stationi %d -",stationi);//stationi==294で落ちる
        return;
 }
+