blob: d0af541fb11a2d121bafda31ae50767d60f78cc9 [file] [log] [blame]
/*
* Copyright (C) 2016 Metrological Group B.V.
* Copyright (C) 2016, 2017, 2018 Igalia S.L
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* aint with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "MediaSampleGStreamer.h"
#include "GStreamerUtilities.h"
#if ENABLE(VIDEO) && USE(GSTREAMER)
namespace WebCore {
MediaSampleGStreamer::MediaSampleGStreamer(GRefPtr<GstSample>&& sample, const FloatSize& presentationSize, const AtomicString& trackId)
: m_pts(MediaTime::zeroTime())
, m_dts(MediaTime::zeroTime())
, m_duration(MediaTime::zeroTime())
, m_trackId(trackId)
, m_presentationSize(presentationSize)
{
ASSERT(sample);
GstBuffer* buffer = gst_sample_get_buffer(sample.get());
if (!buffer)
return;
auto createMediaTime =
[](GstClockTime time) -> MediaTime {
return MediaTime(GST_TIME_AS_USECONDS(time), G_USEC_PER_SEC);
};
if (GST_BUFFER_PTS_IS_VALID(buffer))
m_pts = createMediaTime(GST_BUFFER_PTS(buffer));
if (GST_BUFFER_DTS_IS_VALID(buffer) || GST_BUFFER_PTS_IS_VALID(buffer))
m_dts = createMediaTime(GST_BUFFER_DTS_OR_PTS(buffer));
if (GST_BUFFER_DURATION_IS_VALID(buffer))
m_duration = createMediaTime(GST_BUFFER_DURATION(buffer));
m_size = gst_buffer_get_size(buffer);
m_sample = sample;
if (GST_BUFFER_FLAG_IS_SET(buffer, GST_BUFFER_FLAG_DELTA_UNIT))
m_flags = MediaSample::None;
if (GST_BUFFER_FLAG_IS_SET(buffer, GST_BUFFER_FLAG_DECODE_ONLY))
m_flags = static_cast<MediaSample::SampleFlags>(m_flags | MediaSample::IsNonDisplaying);
}
MediaSampleGStreamer::MediaSampleGStreamer(const FloatSize& presentationSize, const AtomicString& trackId)
: m_pts(MediaTime::zeroTime())
, m_dts(MediaTime::zeroTime())
, m_duration(MediaTime::zeroTime())
, m_trackId(trackId)
, m_presentationSize(presentationSize)
{
}
Ref<MediaSampleGStreamer> MediaSampleGStreamer::createFakeSample(GstCaps*, MediaTime pts, MediaTime dts, MediaTime duration, const FloatSize& presentationSize, const AtomicString& trackId)
{
MediaSampleGStreamer* gstreamerMediaSample = new MediaSampleGStreamer(presentationSize, trackId);
gstreamerMediaSample->m_pts = pts;
gstreamerMediaSample->m_dts = dts;
gstreamerMediaSample->m_duration = duration;
gstreamerMediaSample->m_flags = MediaSample::IsNonDisplaying;
return adoptRef(*gstreamerMediaSample);
}
void MediaSampleGStreamer::applyPtsOffset(MediaTime timestampOffset)
{
if (m_pts > timestampOffset) {
m_duration = m_duration + (m_pts - timestampOffset);
m_pts = timestampOffset;
}
}
void MediaSampleGStreamer::offsetTimestampsBy(const MediaTime& timestampOffset)
{
if (!timestampOffset)
return;
m_pts += timestampOffset;
m_dts += timestampOffset;
if (auto* buffer = gst_sample_get_buffer(m_sample.get())) {
GST_BUFFER_PTS(buffer) = toGstClockTime(m_pts);
GST_BUFFER_DTS(buffer) = toGstClockTime(m_dts);
}
}
PlatformSample MediaSampleGStreamer::platformSample()
{
PlatformSample sample = { PlatformSample::GStreamerSampleType, { .gstSample = m_sample.get() } };
return sample;
}
Ref<MediaSample> MediaSampleGStreamer::createNonDisplayingCopy() const
{
if (!m_sample)
return createFakeSample(nullptr, m_pts, m_dts, m_duration, m_presentationSize, m_trackId);
GstBuffer* buffer = gst_sample_get_buffer(m_sample.get());
GST_BUFFER_FLAG_SET(buffer, GST_BUFFER_FLAG_DECODE_ONLY);
GstCaps* caps = gst_sample_get_caps(m_sample.get());
GstSegment* segment = gst_sample_get_segment(m_sample.get());
const GstStructure* originalInfo = gst_sample_get_info(m_sample.get());
GstStructure* info = originalInfo ? gst_structure_copy(originalInfo) : nullptr;
GRefPtr<GstSample> sample = adoptGRef(gst_sample_new(buffer, caps, segment, info));
return adoptRef(*new MediaSampleGStreamer(sample.get(), m_presentationSize, m_trackId));
}
void MediaSampleGStreamer::dump(PrintStream& out) const
{
out.print("{PTS(", presentationTime(), "), DTS(", decodeTime(), "), duration(", duration(), "), flags(");
bool anyFlags = false;
auto appendFlag = [&out, &anyFlags](const char* flagName) {
if (anyFlags)
out.print(",");
out.print(flagName);
anyFlags = true;
};
if (flags() & MediaSample::IsSync)
appendFlag("sync");
if (flags() & MediaSample::IsNonDisplaying)
appendFlag("non-displaying");
if (flags() & MediaSample::HasAlpha)
appendFlag("has-alpha");
if (flags() & ~(MediaSample::IsSync | MediaSample::IsNonDisplaying | MediaSample::HasAlpha))
appendFlag("unknown-flag");
out.print("), trackId(", trackID().string(), "), presentationSize(", presentationSize().width(), "x", presentationSize().height(), ")}");
}
} // namespace WebCore.
#endif // ENABLE(VIDEO) && USE(GSTREAMER)