void vcdplayer_set_origin(access_t *p_access, lsn_t i_lsn, track_t i_track, const vcdinfo_itemid_t *p_itemid) { vcdplayer_t *p_vcdplayer = (vcdplayer_t *)p_access->p_sys; const size_t i_size= vcdplayer_get_item_size(p_access, *p_itemid); if( VCDINFO_NULL_LSN == i_lsn ) { LOG_ERR("%s %d", "Invalid LSN for track", i_track); return; } p_vcdplayer->play_item.num = p_itemid->num; p_vcdplayer->play_item.type = p_itemid->type; p_vcdplayer->i_lsn = i_lsn; p_vcdplayer->end_lsn = p_vcdplayer->i_lsn + i_size; p_vcdplayer->origin_lsn = p_vcdplayer->i_lsn; p_vcdplayer->i_track = i_track; p_vcdplayer->track_lsn = vcdinfo_get_track_lsn(p_vcdplayer->vcd, i_track); p_vcdplayer->track_end_lsn = p_vcdplayer->track_lsn + vcdinfo_get_track_sect_count(p_vcdplayer->vcd, i_track); dbg_print((INPUT_DBG_CALL|INPUT_DBG_LSN), "lsn %u, end LSN: %u item.num %d, item.type %d", p_vcdplayer->i_lsn, p_vcdplayer->end_lsn, p_vcdplayer->play_item.num, p_vcdplayer->play_item.type); }
/***************************************************************************** VCDEntryPoints: Reads the information about the entry points on the disc and initializes area information with that. Before calling this track information should have been read in. *****************************************************************************/ static bool VCDEntryPoints( access_t * p_access ) { if (!p_access || !p_access->p_sys) return false; vcdplayer_t *p_vcdplayer = (vcdplayer_t *) p_access->p_sys; const unsigned int i_entries = vcdinfo_get_num_entries(p_vcdplayer->vcd); const track_t i_last_track = cdio_get_num_tracks(vcdinfo_get_cd_image(p_vcdplayer->vcd)) + cdio_get_first_track_num(vcdinfo_get_cd_image(p_vcdplayer->vcd)); unsigned int i; if (0 == i_entries) { LOG_ERR ("no entires found -- something is wrong" ); return false; } p_vcdplayer->p_entries = malloc( sizeof( lsn_t ) * i_entries ); if( p_vcdplayer->p_entries == NULL ) { LOG_ERR ("not enough memory for entry points treatment" ); return false; } p_vcdplayer->i_entries = i_entries; for( i = 0 ; i < i_entries ; i++ ) { const track_t i_track = vcdinfo_get_track(p_vcdplayer->vcd, i); if( i_track <= i_last_track ) { seekpoint_t *s = vlc_seekpoint_New(); char psz_entry[100]; snprintf(psz_entry, sizeof(psz_entry), "%s %02d", _("Entry"), i ); p_vcdplayer->p_entries[i] = vcdinfo_get_entry_lsn(p_vcdplayer->vcd, i); s->psz_name = strdup(psz_entry); s->i_byte_offset = (p_vcdplayer->p_entries[i] - vcdinfo_get_track_lsn(p_vcdplayer->vcd,i_track)) * M2F2_SECTOR_SIZE; dbg_print( INPUT_DBG_MRL, "%s, lsn %d, byte_offset %"PRId64"", s->psz_name, p_vcdplayer->p_entries[i], s->i_byte_offset); TAB_APPEND( p_vcdplayer->p_title[i_track-1]->i_seekpoint, p_vcdplayer->p_title[i_track-1]->seekpoint, s ); } else msg_Warn( p_access, "wrong track number found in entry points" ); } p_vcdplayer->b_valid_ep = true; return true; }
/*! Set reading to play an entire track. */ static void _vcdplayer_set_track(access_t * p_access, track_t i_track) { vcdplayer_t *p_vcdplayer = (vcdplayer_t *)p_access->p_sys; if (i_track < 1 || i_track > p_vcdplayer->i_tracks) return; else { const vcdinfo_obj_t *p_vcdinfo = p_vcdplayer->vcd; vcdinfo_itemid_t itemid; itemid.num = i_track; itemid.type = VCDINFO_ITEM_TYPE_TRACK; p_vcdplayer->in_still = 0; VCDSetOrigin(p_access, vcdinfo_get_track_lsn(p_vcdinfo, i_track), i_track, &itemid); dbg_print(INPUT_DBG_LSN, "LSN: %u", p_vcdplayer->i_lsn); } }
/***************************************************************************** * vcd_Open: Opens a VCD device or file initializes, a list of tracks, segements and entry lsns and sizes and returns an opaque handle. *****************************************************************************/ static vcdinfo_obj_t * vcd_Open( vlc_object_t *p_this, const char *psz_dev ) { access_t *p_access = (access_t *)p_this; vcdplayer_t *p_vcdplayer = (vcdplayer_t *) p_access->p_sys; vcdinfo_obj_t *p_vcdobj; char *actual_dev; unsigned int i; dbg_print(INPUT_DBG_CALL, "called with %s", psz_dev); if( !psz_dev ) return NULL; actual_dev= ToLocaleDup(psz_dev); if( vcdinfo_open(&p_vcdobj, &actual_dev, DRIVER_UNKNOWN, NULL) != VCDINFO_OPEN_VCD) { free(actual_dev); return NULL; } free(actual_dev); /* Save summary info on tracks, segments and entries... */ if ( 0 < (p_vcdplayer->i_tracks = vcdinfo_get_num_tracks(p_vcdobj)) ) { p_vcdplayer->track = (vcdplayer_play_item_info_t *) calloc(p_vcdplayer->i_tracks, sizeof(vcdplayer_play_item_info_t)); for (i=0; i<p_vcdplayer->i_tracks; i++) { unsigned int track_num=i+1; p_vcdplayer->track[i].size = vcdinfo_get_track_sect_count(p_vcdobj, track_num); p_vcdplayer->track[i].start_LSN = vcdinfo_get_track_lsn(p_vcdobj, track_num); } } else p_vcdplayer->track = NULL; if( 0 < (p_vcdplayer->i_entries = vcdinfo_get_num_entries(p_vcdobj)) ) { p_vcdplayer->entry = (vcdplayer_play_item_info_t *) calloc(p_vcdplayer->i_entries, sizeof(vcdplayer_play_item_info_t)); for (i=0; i<p_vcdplayer->i_entries; i++) { p_vcdplayer->entry[i].size = vcdinfo_get_entry_sect_count(p_vcdobj, i); p_vcdplayer->entry[i].start_LSN = vcdinfo_get_entry_lsn(p_vcdobj, i); } } else p_vcdplayer->entry = NULL; if ( 0 < (p_vcdplayer->i_segments = vcdinfo_get_num_segments(p_vcdobj)) ) { p_vcdplayer->segment = (vcdplayer_play_item_info_t *) calloc(p_vcdplayer->i_segments, sizeof(vcdplayer_play_item_info_t)); for (i=0; i<p_vcdplayer->i_segments; i++) { p_vcdplayer->segment[i].size = vcdinfo_get_seg_sector_count(p_vcdobj, i); p_vcdplayer->segment[i].start_LSN = vcdinfo_get_seg_lsn(p_vcdobj, i); } } else p_vcdplayer->segment = NULL; return p_vcdobj; }
/* Sets start origin for subsequent seeks/reads */ void VCDSetOrigin( access_t *p_access, lsn_t i_lsn, track_t i_track, const vcdinfo_itemid_t *p_itemid ) { vcdplayer_t *p_vcdplayer= (vcdplayer_t *)p_access->p_sys; dbg_print( (INPUT_DBG_CALL|INPUT_DBG_LSN), "i_lsn: %lu, track: %d", (long unsigned int) i_lsn, i_track ); vcdplayer_set_origin(p_access, i_lsn, i_track, p_itemid); switch (p_vcdplayer->play_item.type) { case VCDINFO_ITEM_TYPE_ENTRY: VCDUpdateVar( p_access, p_itemid->num, VLC_VAR_SETVALUE, "chapter", _("Entry"), "Setting entry/segment"); p_access->info.i_title = i_track-1; if (p_vcdplayer->b_track_length) { p_vcdplayer->size = p_vcdplayer->p_title[i_track-1]->i_size; p_access->info.i_pos = (uint64_t) M2F2_SECTOR_SIZE * (vcdinfo_get_track_lsn(p_vcdplayer->vcd, i_track)-i_lsn); } else { p_vcdplayer->size = M2F2_SECTOR_SIZE * (int64_t) vcdinfo_get_entry_sect_count(p_vcdplayer->vcd,p_itemid->num); p_access->info.i_pos = 0; } dbg_print( (INPUT_DBG_LSN|INPUT_DBG_PBC), "size: %"PRIu64", pos: %"PRIu64, p_vcdplayer->size, p_access->info.i_pos ); p_access->info.i_seekpoint = p_itemid->num; break; case VCDINFO_ITEM_TYPE_SEGMENT: VCDUpdateVar( p_access, p_itemid->num, VLC_VAR_SETVALUE, "chapter", _("Segment"), "Setting entry/segment"); /* The last title entry is the for segments (when segments exist and they must here. The segment seekpoints are stored after the entry seekpoints and (zeroed) lid seekpoints. */ p_access->info.i_title = p_vcdplayer->i_titles - 1; p_vcdplayer->size = 0; /* No seeking on stills, please. */ p_access->info.i_pos = 0; p_access->info.i_seekpoint = p_vcdplayer->i_entries + p_vcdplayer->i_lids + p_itemid->num; break; case VCDINFO_ITEM_TYPE_TRACK: p_access->info.i_title = i_track-1; p_vcdplayer->size = p_vcdplayer->p_title[i_track-1]->i_size; p_access->info.i_pos = 0; p_access->info.i_seekpoint = vcdinfo_track_get_entry(p_vcdplayer->vcd, i_track); break; default: msg_Warn( p_access, "can't set origin for play type %d", p_vcdplayer->play_item.type ); } p_access->info.i_update = INPUT_UPDATE_TITLE|INPUT_UPDATE_SEEKPOINT; VCDUpdateTitle( p_access ); }