ext/resindvd/resindvdsrc.c: Additional patch for resin fixes time-based seeking in titles that don't start at sector ...

Original commit message from CVS:
* ext/resindvd/resindvdsrc.c:
Additional patch for resin fixes time-based seeking in titles
that don't start at sector 0 in the VTS, and fixes interpretation
of the cell elapsed time, so that it reports the correct time after
the seek completes.
Really fixes #566957.
This commit is contained in:
Jan Schmidt 2009-01-09 16:49:57 +00:00
parent c266d310af
commit d119d31259
2 changed files with 101 additions and 21 deletions

View file

@ -1,3 +1,12 @@
2009-01-09 Jan Schmidt <jan.schmidt@sun.com>
* ext/resindvd/resindvdsrc.c:
Additional patch for resin fixes time-based seeking in titles
that don't start at sector 0 in the VTS, and fixes interpretation
of the cell elapsed time, so that it reports the correct time after
the seek completes.
Really fixes #566957.
2009-01-08 Wim Taymans <wim.taymans@collabora.co.uk> 2009-01-08 Wim Taymans <wim.taymans@collabora.co.uk>
Based on Patch by: Brian Cameron <brian dot cameron at sun dot com> Based on Patch by: Brian Cameron <brian dot cameron at sun dot com>

View file

@ -146,9 +146,12 @@ ifotime_to_gsttime (dvd_time_t * ifo_time)
GstClockTime ts; GstClockTime ts;
guint frames; guint frames;
ts = 3600 * GST_SECOND * ifo_time->hour; ts = 36000 * GST_SECOND * ((ifo_time->hour & 0xf0) >> 4);
ts += 60 * GST_SECOND * ifo_time->minute; ts += 3600 * GST_SECOND * (ifo_time->hour & 0x0f);
ts += GST_SECOND * ifo_time->second; ts += 600 * GST_SECOND * ((ifo_time->minute & 0xf0) >> 4);
ts += 60 * GST_SECOND * (ifo_time->minute & 0x0f);
ts += 10 * GST_SECOND * ((ifo_time->second & 0xf0) >> 4);
ts += GST_SECOND * (ifo_time->second & 0x0f);
frames = ((ifo_time->frame_u >> 4) & 0x3) * 10; frames = ((ifo_time->frame_u >> 4) & 0x3) * 10;
frames += (ifo_time->frame_u & 0xf); frames += (ifo_time->frame_u & 0xf);
@ -786,7 +789,7 @@ rsn_dvdsrc_step (resinDvdSrc * src, gboolean have_dvd_lock)
src->vts_file = ifoOpen (src->dvdread, src->vts_n); src->vts_file = ifoOpen (src->dvdread, src->vts_n);
} }
src->in_menu = !dvdnav_is_domain_vtsm (src->dvdnav); src->in_menu = !dvdnav_is_domain_vts (src->dvdnav);
if (!dvdnav_is_domain_fp (src->dvdnav)) if (!dvdnav_is_domain_fp (src->dvdnav))
rsn_dvdsrc_prepare_streamsinfo_event (src); rsn_dvdsrc_prepare_streamsinfo_event (src);
@ -1614,7 +1617,7 @@ rsn_dvdsrc_nav_clock_cb (GstClock * clock, GstClockTime time, GstClockID id,
if (time < base_time + cur->running_ts) if (time < base_time + cur->running_ts)
break; /* Next NAV is in the future */ break; /* Next NAV is in the future */
GST_DEBUG_OBJECT (src, "Activating nav pack for with TS %" GST_TIME_FORMAT GST_DEBUG_OBJECT (src, "Activating nav pack with TS %" GST_TIME_FORMAT
" at running TS %" GST_TIME_FORMAT, GST_TIME_ARGS (cur->ts), " at running TS %" GST_TIME_FORMAT, GST_TIME_ARGS (cur->ts),
GST_TIME_ARGS (cur->running_ts)); GST_TIME_ARGS (cur->running_ts));
rsn_dvdsrc_activate_nav_block (src, cur->buffer); rsn_dvdsrc_activate_nav_block (src, cur->buffer);
@ -1709,6 +1712,16 @@ rsn_dvdsrc_src_event (RsnBaseSrc * basesrc, GstEvent * event)
case GST_EVENT_NAVIGATION: case GST_EVENT_NAVIGATION:
res = rsn_dvdsrc_handle_navigation_event (src, event); res = rsn_dvdsrc_handle_navigation_event (src, event);
break; break;
case GST_EVENT_SEEK:{
GstSeekFlags flags;
gst_event_parse_seek (event, NULL, NULL, &flags, NULL, NULL, NULL, NULL);
src->flushing_seek = !!(flags & GST_SEEK_FLAG_FLUSH);
GST_DEBUG_OBJECT (src, "%s seek event",
src->flushing_seek ? "flushing" : "non-flushing");
res = GST_BASE_SRC_CLASS (parent_class)->event (basesrc, event);
break;
}
default: default:
res = GST_BASE_SRC_CLASS (parent_class)->event (basesrc, event); res = GST_BASE_SRC_CLASS (parent_class)->event (basesrc, event);
break; break;
@ -1854,17 +1867,21 @@ static gboolean
rsn_dvdsrc_prepare_seek (RsnBaseSrc * bsrc, GstEvent * event, rsn_dvdsrc_prepare_seek (RsnBaseSrc * bsrc, GstEvent * event,
GstSegment * segment) GstSegment * segment)
{ {
resinDvdSrc *src = RESINDVDSRC (bsrc);
GstSeekType cur_type, stop_type; GstSeekType cur_type, stop_type;
gint64 cur, stop; gint64 cur, stop;
GstSeekFlags flags; GstSeekFlags flags;
GstFormat seek_format; GstFormat seek_format;
gdouble rate; gdouble rate;
gboolean update; gboolean update;
gboolean ret;
gst_event_parse_seek (event, &rate, &seek_format, &flags, gst_event_parse_seek (event, &rate, &seek_format, &flags,
&cur_type, &cur, &stop_type, &stop); &cur_type, &cur, &stop_type, &stop);
/* Don't allow bytes seeks - angle, time, chapter, title only is the plan */
if (seek_format == GST_FORMAT_BYTES)
return FALSE;
if (seek_format == rsndvd_format || seek_format == title_format || if (seek_format == rsndvd_format || seek_format == title_format ||
seek_format == chapter_format) { seek_format == chapter_format) {
/* Seeks in our internal formats are passed directly through to the do_seek /* Seeks in our internal formats are passed directly through to the do_seek
@ -1873,18 +1890,14 @@ rsn_dvdsrc_prepare_seek (RsnBaseSrc * bsrc, GstEvent * event,
gst_segment_set_seek (segment, rate, seek_format, flags, cur_type, cur, gst_segment_set_seek (segment, rate, seek_format, flags, cur_type, cur,
stop_type, stop, &update); stop_type, stop, &update);
if (flags & GST_SEEK_FLAG_FLUSH)
src->flushing_seek = TRUE;
return TRUE; return TRUE;
} }
/* Don't allow bytes seeks - angle, time, chapter, title only is the plan */
if (seek_format == GST_FORMAT_BYTES)
return FALSE;
/* Let basesrc handle other formats for now. FIXME: Implement angle */ /* Let basesrc handle other formats for now. FIXME: Implement angle */
return GST_BASE_SRC_CLASS (parent_class)->prepare_seek_segment (bsrc, ret = GST_BASE_SRC_CLASS (parent_class)->prepare_seek_segment (bsrc,
event, segment); event, segment);
return ret;
} }
/* Find sector from time using time map if available */ /* Find sector from time using time map if available */
@ -1894,7 +1907,9 @@ rsn_dvdsrc_get_sector_from_time_tmap (resinDvdSrc * src, GstClockTime ts)
vts_tmapt_t *vts_tmapt; vts_tmapt_t *vts_tmapt;
vts_tmap_t *title_tmap; vts_tmap_t *title_tmap;
gint32 title, part, vts_ttn; gint32 title, part, vts_ttn;
guint32 entry; guint32 entry, sector, logical_sector;
gint pgc_n, cell_n;
pgc_t *pgc;
if (ts == 0) if (ts == 0)
return 0; return 0;
@ -1906,27 +1921,80 @@ rsn_dvdsrc_get_sector_from_time_tmap (resinDvdSrc * src, GstClockTime ts)
DVDNAV_STATUS_OK) DVDNAV_STATUS_OK)
return -1; return -1;
vts_tmapt = src->vts_file->vts_tmapt;
if (vts_tmapt == NULL)
return -1;
/* To find the right tmap, we need the title number within this VTS (vts_ttn) /* To find the right tmap, we need the title number within this VTS (vts_ttn)
* from the VMG tt_srpt table... */ * from the VMG tt_srpt table... */
if (title < 1 || title >= src->vmg_file->tt_srpt->nr_of_srpts) if (title < 1 || title >= src->vmg_file->tt_srpt->nr_of_srpts)
return -1; return -1;
vts_tmapt = src->vts_file->vts_tmapt; /* We must be in the correct VTS for any of this to succeed... */
if (vts_tmapt == NULL) if (src->vts_n != src->vmg_file->tt_srpt->title[title - 1].title_set_nr)
return -1;
/* We must also be in the VTS domain to use the tmap table */
if (src->vts_n == 0 || src->in_menu)
return -1; return -1;
vts_ttn = src->vmg_file->tt_srpt->title[title - 1].vts_ttn; vts_ttn = src->vmg_file->tt_srpt->title[title - 1].vts_ttn;
GST_DEBUG_OBJECT (src, "Seek to time %" GST_TIME_FORMAT
" in VTS %d title %d (vts_ttn %d of %d)",
GST_TIME_ARGS (ts), src->vts_n, title, vts_ttn, vts_tmapt->nr_of_tmaps);
if (vts_ttn < 1 || vts_ttn > vts_tmapt->nr_of_tmaps) if (vts_ttn < 1 || vts_ttn > vts_tmapt->nr_of_tmaps)
return -1; return -1;
/* We'll be needing to look up the PGC later, so it better exist */
if (vts_ttn > src->vts_file->vts_ptt_srpt->nr_of_srpts)
return -1;
title_tmap = vts_tmapt->tmap + vts_ttn - 1; title_tmap = vts_tmapt->tmap + vts_ttn - 1;
entry = ts / (title_tmap->tmu * GST_SECOND); entry = ts / (title_tmap->tmu * GST_SECOND);
if (entry == 0)
return 0;
if (entry < title_tmap->nr_of_entries) if (entry < 1 || entry > title_tmap->nr_of_entries)
return title_tmap->map_ent[entry] & 0x7fffffff; return -1;
return -1; sector = title_tmap->map_ent[entry - 1] & 0x7fffffff;
GST_LOG_OBJECT (src, "Got sector %u for time seek (entry %d of %d)\n",
sector, entry, title_tmap->nr_of_entries);
/* Sector is now an absolute sector within the current VTS, but
* dvdnav_sector_search expects a logical sector within the current PGC...
* which means iterating over the cells of the current PGC until we find
* the cell that contains the time and sector we want, accumulating
* the logical sector offsets until we find it
*/
if (src->vts_file->vts_ptt_srpt->title[vts_ttn - 1].nr_of_ptts == 0)
return -1;
pgc_n = src->vts_file->vts_ptt_srpt->title[vts_ttn - 1].ptt[0].pgcn;
if (pgc_n > src->vts_file->vts_pgcit->nr_of_pgci_srp)
return -1;
pgc = src->vts_file->vts_pgcit->pgci_srp[pgc_n - 1].pgc;
logical_sector = 0;
for (cell_n = 0; cell_n < pgc->nr_of_cells; cell_n++) {
cell_playback_t *cell = pgc->cell_playback + cell_n;
if (sector >= cell->first_sector && sector <= cell->last_sector) {
logical_sector += sector - cell->first_sector;
break;
}
logical_sector += (cell->last_sector - cell->first_sector + 1);
}
GST_DEBUG_OBJECT (src, "Mapped sector %u onto PGC relative sector %u\n",
sector, logical_sector);
return logical_sector;
} }
/* call with DVD lock held */ /* call with DVD lock held */
@ -1969,6 +2037,7 @@ rsn_dvdsrc_do_seek (RsnBaseSrc * bsrc, GstSegment * segment)
/* HACK to make initial seek work: */ /* HACK to make initial seek work: */
if (segment->format == GST_FORMAT_TIME) { if (segment->format == GST_FORMAT_TIME) {
g_mutex_lock (src->dvd_lock); g_mutex_lock (src->dvd_lock);
src->discont = TRUE;
ret = rsn_dvdsrc_seek_to_time (src, segment->start); ret = rsn_dvdsrc_seek_to_time (src, segment->start);
g_mutex_unlock (src->dvd_lock); g_mutex_unlock (src->dvd_lock);
} else if (segment->format == title_format) { } else if (segment->format == title_format) {
@ -2055,8 +2124,10 @@ rsn_dvdsrc_do_seek (RsnBaseSrc * bsrc, GstSegment * segment)
segment->duration = -1; segment->duration = -1;
g_print ("seek completed. New start TS %" GST_TIME_FORMAT g_print ("seek completed. New start TS %" GST_TIME_FORMAT
" pos %" GST_TIME_FORMAT "\n", GST_TIME_ARGS (segment->start), " pos %" GST_TIME_FORMAT " (offset %" GST_TIME_FORMAT ")\n",
GST_TIME_ARGS (segment->time)); GST_TIME_ARGS (segment->start),
GST_TIME_ARGS (segment->time),
GST_TIME_ARGS ((GstClockTimeDiff) (segment->start - segment->time)));
src->need_segment = FALSE; src->need_segment = FALSE;
} }