/* * Copyright (C) 2005-2013 Team XBMC * http://xbmc.org * * 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, 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 XBMC; see the file COPYING. If not, see * . * */ #include "DVDDemuxVobsub.h" #include "DVDInputStreams/DVDFactoryInputStream.h" #include "DVDInputStreams/DVDInputStream.h" #include "DVDStreamInfo.h" #include "DVDCodecs/DVDCodecs.h" #include "DVDDemuxers/DVDDemuxFFmpeg.h" #include "DVDDemuxers/DVDDemuxUtils.h" #include "DVDClock.h" #include "DVDSubtitles/DVDSubtitleStream.h" #include using namespace std; CDVDDemuxVobsub::CDVDDemuxVobsub() { } CDVDDemuxVobsub::~CDVDDemuxVobsub() { for(unsigned i=0;i pStream(new CDVDSubtitleStream()); if(!pStream->Open(filename)) return false; string vobsub = subfilename; if ( vobsub == "") { vobsub = filename; vobsub.erase(vobsub.rfind('.'), vobsub.size()); vobsub += ".sub"; } m_Input.reset(CDVDFactoryInputStream::CreateInputStream(NULL, vobsub, "")); if(!m_Input.get() || !m_Input->Open(vobsub.c_str(), "video/x-vobsub")) return false; m_Demuxer.reset(new CDVDDemuxFFmpeg()); if(!m_Demuxer->Open(m_Input.get())) return false; CDVDStreamInfo hints; CDVDCodecOptions options; hints.codec = AV_CODEC_ID_DVD_SUBTITLE; char line[2048]; DECLARE_UNUSED(bool,res) SState state; state.delay = 0; state.id = -1; while( pStream->ReadLine(line, sizeof(line)) ) { if (*line == 0 || *line == '\r' || *line == '\n' || *line == '#') continue; else if (strncmp("langidx:", line, 8) == 0) res = ParseLangIdx(state, line + 8); else if (strncmp("delay:", line, 6) == 0) res = ParseDelay(state, line + 6); else if (strncmp("id:", line, 3) == 0) res = ParseId(state, line + 3); else if (strncmp("timestamp:", line, 10) == 0) res = ParseTimestamp(state, line + 10); else if (strncmp("palette:", line, 8) == 0 || strncmp("size:", line, 5) == 0 || strncmp("org:", line, 4) == 0 || strncmp("custom colors:", line, 14) == 0 || strncmp("scale:", line, 6) == 0 || strncmp("alpha:", line, 6) == 0 || strncmp("fadein/out:", line, 11) == 0 || strncmp("forced subs:", line, 12) == 0) res = ParseExtra(state, line); else continue; } struct sorter s; sort(m_Timestamps.begin(), m_Timestamps.end(), s); m_Timestamp = m_Timestamps.begin(); for(unsigned i=0;iExtraSize = state.extra.length()+1; m_Streams[i]->ExtraData = new uint8_t[m_Streams[i]->ExtraSize]; strcpy((char*)m_Streams[i]->ExtraData, state.extra.c_str()); } return true; } void CDVDDemuxVobsub::Reset() { Flush(); } void CDVDDemuxVobsub::Flush() { m_Demuxer->Flush(); } bool CDVDDemuxVobsub::SeekTime(int time, bool backwords, double* startpts) { double pts = DVD_MSEC_TO_TIME(time); m_Timestamp = m_Timestamps.begin(); for(;m_Timestamp != m_Timestamps.end();++m_Timestamp) { if(m_Timestamp->pts > pts) break; } for(unsigned i=0;i::iterator current; do { if(m_Timestamp == m_Timestamps.end()) return NULL; current = m_Timestamp++; } while(m_Streams[current->id]->m_discard == AVDISCARD_ALL); if(!m_Demuxer->SeekByte(current->pos)) return NULL; DemuxPacket *packet = m_Demuxer->Read(); if(!packet) return NULL; packet->iStreamId = current->id; packet->pts = current->pts; packet->dts = current->pts; return packet; } bool CDVDDemuxVobsub::ParseLangIdx(SState& state, char* line) { return true; } bool CDVDDemuxVobsub::ParseDelay(SState& state, char* line) { int h,m,s,ms; bool negative = false; while(*line == ' ') line++; if(*line == '-') { line++; negative = true; } if(sscanf(line, "%d:%d:%d:%d", &h, &m, &s, &ms) != 4) return false; state.delay = h*3600.0 + m*60.0 + s + ms*0.001; if(negative) state.delay *= -1; return true; } bool CDVDDemuxVobsub::ParseId(SState& state, char* line) { unique_ptr stream(new CStream(this)); while(*line == ' ') line++; strncpy(stream->language, line, 2); stream->language[2] = '\0'; line+=2; while(*line == ' ' || *line == ',') line++; if (strncmp("index:", line, 6) == 0) { line+=6; while(*line == ' ') line++; stream->iPhysicalId = atoi(line); } else stream->iPhysicalId = -1; stream->codec = AV_CODEC_ID_DVD_SUBTITLE; stream->iId = m_Streams.size(); stream->source = STREAM_SOURCE_DEMUX_SUB; state.id = stream->iId; m_Streams.push_back(stream.release()); return true; } bool CDVDDemuxVobsub::ParseExtra(SState& state, char* line) { state.extra += line; state.extra += '\n'; return true; } bool CDVDDemuxVobsub::ParseTimestamp(SState& state, char* line) { if(state.id < 0) return false; int h,m,s,ms; STimestamp timestamp; while(*line == ' ') line++; if(sscanf(line, "%d:%d:%d:%d, filepos:%" PRIx64, &h, &m, &s, &ms, ×tamp.pos) != 5) return false; timestamp.id = state.id; timestamp.pts = DVD_SEC_TO_TIME(state.delay + h*3600.0 + m*60.0 + s + ms*0.001); m_Timestamps.push_back(timestamp); return true; }