2 * Copyright (C) 2017 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "DisplayCaptureSourceCocoa.h"
29 #if ENABLE(MEDIA_STREAM)
32 #include "RealtimeMediaSource.h"
33 #include "RealtimeMediaSourceCenter.h"
34 #include "RealtimeMediaSourceSettings.h"
36 #include <CoreMedia/CMSync.h>
37 #include <mach/mach_time.h>
38 #include <pal/avfoundation/MediaTimeAVFoundation.h>
39 #include <pal/cf/CoreMediaSoftLink.h>
40 #include <pal/spi/cf/CoreAudioSPI.h>
42 #include <wtf/MainThread.h>
43 #include <wtf/NeverDestroyed.h>
48 DisplayCaptureSourceCocoa::DisplayCaptureSourceCocoa(const String& name)
49 : RealtimeMediaSource("", Type::Video, name)
50 , m_timer(RunLoop::current(), this, &DisplayCaptureSourceCocoa::emitFrame)
54 DisplayCaptureSourceCocoa::~DisplayCaptureSourceCocoa()
57 RealtimeMediaSourceCenter::singleton().videoFactory().unsetActiveSource(*this);
61 const RealtimeMediaSourceCapabilities& DisplayCaptureSourceCocoa::capabilities() const
63 if (!m_capabilities) {
64 RealtimeMediaSourceCapabilities capabilities(settings().supportedConstraints());
66 // FIXME: what should these be?
67 capabilities.setWidth(CapabilityValueOrRange(72, 2880));
68 capabilities.setHeight(CapabilityValueOrRange(45, 1800));
69 capabilities.setFrameRate(CapabilityValueOrRange(.01, 60.0));
71 m_capabilities = WTFMove(capabilities);
73 return m_capabilities.value();
76 const RealtimeMediaSourceSettings& DisplayCaptureSourceCocoa::settings() const
78 if (!m_currentSettings) {
79 RealtimeMediaSourceSettings settings;
80 settings.setFrameRate(frameRate());
81 auto size = this->size();
82 if (size.width() && size.height()) {
83 settings.setWidth(size.width());
84 settings.setHeight(size.height());
87 RealtimeMediaSourceSupportedConstraints supportedConstraints;
88 supportedConstraints.setSupportsFrameRate(true);
89 supportedConstraints.setSupportsWidth(true);
90 supportedConstraints.setSupportsHeight(true);
91 supportedConstraints.setSupportsAspectRatio(true);
92 settings.setSupportedConstraints(supportedConstraints);
94 m_currentSettings = WTFMove(settings);
96 return m_currentSettings.value();
99 void DisplayCaptureSourceCocoa::settingsDidChange()
101 m_currentSettings = std::nullopt;
102 RealtimeMediaSource::settingsDidChange();
105 void DisplayCaptureSourceCocoa::startProducingData()
108 RealtimeMediaSourceCenter::singleton().videoFactory().setActiveSource(*this);
111 m_startTime = MonotonicTime::now();
112 m_timer.startRepeating(1_ms * lround(1000 / frameRate()));
115 void DisplayCaptureSourceCocoa::stopProducingData()
118 m_elapsedTime += MonotonicTime::now() - m_startTime;
119 m_startTime = MonotonicTime::nan();
122 Seconds DisplayCaptureSourceCocoa::elapsedTime()
124 if (std::isnan(m_startTime))
125 return m_elapsedTime;
127 return m_elapsedTime + (MonotonicTime::now() - m_startTime);
130 bool DisplayCaptureSourceCocoa::applySize(const IntSize& newSize)
132 if (size() == newSize)
135 m_bufferAttributes = nullptr;
139 bool DisplayCaptureSourceCocoa::applyFrameRate(double rate)
141 if (m_timer.isActive())
142 m_timer.startRepeating(1_ms * lround(1000 / rate));
147 void DisplayCaptureSourceCocoa::emitFrame()
155 RetainPtr<CMSampleBufferRef> DisplayCaptureSourceCocoa::sampleBufferFromPixelBuffer(CVPixelBufferRef pixelBuffer)
160 CMTime sampleTime = CMTimeMake(((elapsedTime() + 100_ms) * 100).seconds(), 100);
161 CMSampleTimingInfo timingInfo = { kCMTimeInvalid, sampleTime, sampleTime };
163 CMVideoFormatDescriptionRef formatDescription = nullptr;
164 auto status = CMVideoFormatDescriptionCreateForImageBuffer(kCFAllocatorDefault, (CVImageBufferRef)pixelBuffer, &formatDescription);
166 RELEASE_LOG(Media, "Failed to initialize CMVideoFormatDescription with error code: %d", static_cast<int>(status));
170 CMSampleBufferRef sampleBuffer;
171 status = CMSampleBufferCreateReadyWithImageBuffer(kCFAllocatorDefault, (CVImageBufferRef)pixelBuffer, formatDescription, &timingInfo, &sampleBuffer);
172 CFRelease(formatDescription);
174 RELEASE_LOG(Media, "Failed to initialize CMSampleBuffer with error code: %d", static_cast<int>(status));
178 return adoptCF(sampleBuffer);
182 static int32_t roundUpToMacroblockMultiple(int32_t size)
184 return (size + 15) & ~15;
187 RetainPtr<CVPixelBufferRef> DisplayCaptureSourceCocoa::pixelBufferFromIOSurface(IOSurfaceRef surface)
189 if (!m_bufferAttributes) {
190 m_bufferAttributes = adoptCF(CFDictionaryCreateMutable(nullptr, 0, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks));
192 auto format = IOSurfaceGetPixelFormat(surface);
193 if (format == kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange || format == kCVPixelFormatType_420YpCbCr8BiPlanarFullRange) {
195 // If the width x height isn't a multiple of 16 x 16 and the surface has extra memory in the planes, set pixel buffer attributes to reflect it.
196 auto width = IOSurfaceGetWidth(surface);
197 auto height = IOSurfaceGetHeight(surface);
198 int32_t extendedRight = roundUpToMacroblockMultiple(width) - width;
199 int32_t extendedBottom = roundUpToMacroblockMultiple(height) - height;
201 if ((IOSurfaceGetBytesPerRowOfPlane(surface, 0) >= width + extendedRight)
202 && (IOSurfaceGetBytesPerRowOfPlane(surface, 1) >= width + extendedRight)
203 && (IOSurfaceGetAllocSize(surface) >= (height + extendedBottom) * IOSurfaceGetBytesPerRowOfPlane(surface, 0) * 3 / 2)) {
204 auto cfInt = adoptCF(CFNumberCreate(nullptr, kCFNumberIntType, &extendedRight));
205 CFDictionarySetValue(m_bufferAttributes.get(), kCVPixelBufferExtendedPixelsRightKey, cfInt.get());
206 cfInt = adoptCF(CFNumberCreate(nullptr, kCFNumberIntType, &extendedBottom));
207 CFDictionarySetValue(m_bufferAttributes.get(), kCVPixelBufferExtendedPixelsBottomKey, cfInt.get());
211 CFDictionarySetValue(m_bufferAttributes.get(), kCVPixelBufferOpenGLCompatibilityKey, kCFBooleanTrue);
214 CVPixelBufferRef pixelBuffer;
215 auto status = CVPixelBufferCreateWithIOSurface(kCFAllocatorDefault, surface, m_bufferAttributes.get(), &pixelBuffer);
217 RELEASE_LOG(Media, "CVPixelBufferCreateWithIOSurface failed with error code: %d", static_cast<int>(status));
221 return adoptCF(pixelBuffer);
225 } // namespace WebCore
227 #endif // ENABLE(MEDIA_STREAM)