seek.c
来自「VLC媒体播放程序」· C语言 代码 · 共 331 行
C
331 行
/* seek.c: functions to navigate through DVD. ***************************************************************************** * Copyright (C) 1998-2001 VideoLAN * $Id: seek.c,v 1.3 2002/12/06 16:34:04 sam Exp $ * * Author: St閜hane Borel <stef@via.ecp.fr> * * 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, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#include <stdio.h>#include <stdlib.h>#include <string.h>#include <vlc/vlc.h>#include <vlc/input.h>#ifdef HAVE_UNISTD_H# include <unistd.h>#endif#include <fcntl.h>#include <sys/types.h>#include <sys/stat.h>#include <string.h>#include <errno.h>#ifdef STRNCASECMP_IN_STRINGS_H# include <strings.h>#endif#ifdef GOD_DAMN_DMCA# include "dvdcss.h"#else# include <dvdcss/dvdcss.h>#endif#include "dvd.h"#include "seek.h"#include "ifo.h"#define title \ p_dvd->p_ifo->vts.title_unit.p_title[p_dvd->i_title_id-1].title#define cell p_dvd->p_ifo->vts.cell_infint CellIsInterleaved( thread_dvd_data_t * p_dvd ){ return title.p_cell_play[p_dvd->i_prg_cell].i_category & 0xf000;}int CellPrg2Map( thread_dvd_data_t * p_dvd ){ int i_cell; i_cell = p_dvd->i_map_cell; if( i_cell >= cell.i_cell_nb ) { return -1; } while( ( i_cell < cell.i_cell_nb ) && ( ( title.p_cell_pos[p_dvd->i_prg_cell].i_vob_id != cell.p_cell_map[i_cell].i_vob_id ) || ( title.p_cell_pos[p_dvd->i_prg_cell].i_cell_id != cell.p_cell_map[i_cell].i_cell_id ) ) ) { i_cell++; } if( i_cell >= cell.i_cell_nb ) { return -1; } return i_cell;}int CellAngleOffset( thread_dvd_data_t * p_dvd, int i_prg_cell ){ int i_cell_off; if( i_prg_cell >= title.i_cell_nb ) { return 0; } /* basic handling of angles */ switch( ( ( title.p_cell_play[i_prg_cell].i_category & 0xf000 ) >> 12 ) ) { /* we enter a muli-angle section */ case 0x5: i_cell_off = p_dvd->i_angle - 1; p_dvd->i_angle_cell = 0; break; /* we exit a multi-angle section */ case 0x9: case 0xd: i_cell_off = p_dvd->i_angle_nb - p_dvd->i_angle; break; default: i_cell_off = 0; } return i_cell_off;}int CellFirstSector( thread_dvd_data_t * p_dvd ){ return __MAX( cell.p_cell_map[p_dvd->i_map_cell].i_first_sector, title.p_cell_play[p_dvd->i_prg_cell].i_first_sector );}int CellLastSector( thread_dvd_data_t * p_dvd ){ return __MIN( cell.p_cell_map[p_dvd->i_map_cell].i_last_sector, title.p_cell_play[p_dvd->i_prg_cell].i_last_sector );}int NextCellPrg( thread_dvd_data_t * p_dvd ){ unsigned int i_cell = p_dvd->i_prg_cell; if( p_dvd->i_vts_lb > title.p_cell_play[i_cell].i_last_sector ) { i_cell ++; i_cell += CellAngleOffset( p_dvd, i_cell ); if( i_cell >= title.i_cell_nb ) { return -1; } } return i_cell;}int Lb2CellPrg( thread_dvd_data_t * p_dvd ){ unsigned int i_cell = 0; while( p_dvd->i_vts_lb > title.p_cell_play[i_cell].i_last_sector ) { i_cell ++; i_cell += CellAngleOffset( p_dvd, i_cell ); if( i_cell >= title.i_cell_nb ) { return -1; } } return i_cell;}int Lb2CellMap( thread_dvd_data_t * p_dvd ){ int i_cell = 0; while( p_dvd->i_vts_lb > cell.p_cell_map[i_cell].i_last_sector ) { i_cell ++; if( i_cell >= cell.i_cell_nb ) { return -1; } } return i_cell;}int LbMaxOnce( thread_dvd_data_t * p_dvd ){ int i_block_once = p_dvd->i_last_lb + 1 - p_dvd->i_vts_lb; /* Get the position of the next cell if we're at cell end */ if( i_block_once <= 0 ) { p_dvd->i_map_cell++; p_dvd->i_angle_cell++; p_dvd->i_prg_cell = NextCellPrg( p_dvd ); if( p_dvd->i_prg_cell < 0 ) { /* EOF */ return 0; } p_dvd->i_map_cell = CellPrg2Map( p_dvd ); if( p_dvd->i_map_cell < 0 ) { return 0; } p_dvd->i_vts_lb = CellFirstSector( p_dvd ); p_dvd->i_last_lb = CellLastSector( p_dvd ); p_dvd->i_chapter = NextChapter( p_dvd ); if( p_dvd->i_chapter < 0 ) { return 0; } /* Position the fd pointer on the right address */ if( dvdcss_seek( p_dvd->dvdhandle, p_dvd->i_vts_start + p_dvd->i_vts_lb, DVDCSS_SEEK_MPEG ) < 0 ) {#if 0 intf_ErrMsg( "dvd error: %s", dvdcss_error( p_dvd->dvdhandle ) );#endif return 0; } i_block_once = p_dvd->i_last_lb + 1 - p_dvd->i_vts_lb; } return i_block_once;}int CellPrg2Chapter( thread_dvd_data_t * p_dvd ){ unsigned int i_chapter = 1; unsigned int i_cell = p_dvd->i_prg_cell; if( CellIsInterleaved( p_dvd ) ) { i_cell -= (p_dvd->i_angle - 1); } while( title.chapter_map.pi_start_cell[i_chapter] <= i_cell+1 ) { i_chapter ++; if( i_chapter >= p_dvd->i_chapter_nb ) { return p_dvd->i_chapter_nb; } } return i_chapter;}int NextChapter( thread_dvd_data_t * p_dvd ){ int i_cell = p_dvd->i_prg_cell; if( CellIsInterleaved( p_dvd ) ) { i_cell -= (p_dvd->i_angle - 1); } if( title.chapter_map.pi_start_cell[p_dvd->i_chapter] <= i_cell+1 ) { p_dvd->i_chapter++; if( p_dvd->i_chapter > p_dvd->i_chapter_nb ) { return -1; } p_dvd->b_new_chapter = 1; return p_dvd->i_chapter; } return p_dvd->i_chapter;}int DVDSetChapter( thread_dvd_data_t * p_dvd, unsigned int i_chapter ){ if( i_chapter <= 0 || i_chapter > p_dvd->i_chapter_nb ) { i_chapter = 1; } if( p_dvd->i_chapter != i_chapter ) { /* Find cell index in Program chain for current chapter */ p_dvd->i_prg_cell = title.chapter_map.pi_start_cell[i_chapter-1] - 1; p_dvd->i_prg_cell += CellAngleOffset( p_dvd, p_dvd->i_prg_cell ); if( i_chapter < p_dvd->i_chapter ) { p_dvd->i_map_cell = 0; } p_dvd->i_map_cell = CellPrg2Map( p_dvd ); p_dvd->i_vts_lb = CellFirstSector( p_dvd ); p_dvd->i_last_lb = CellLastSector( p_dvd ); /* Position the fd pointer on the right address */ if( dvdcss_seek( p_dvd->dvdhandle, p_dvd->i_vts_start + p_dvd->i_vts_lb, DVDCSS_SEEK_MPEG ) < 0 ) {#if 0 intf_ErrMsg( "dvd error: %s", dvdcss_error( p_dvd->dvdhandle ) );#endif return -1; }#if 0 intf_WarnMsg( 4, "dvd info: chapter %d prg_cell %d map_cell %d", i_chapter, p_dvd->i_prg_cell, p_dvd->i_map_cell );#endif } return i_chapter;}#undef cell#undef title
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?