mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 23:13:18 +00:00
d56ce7399f
- Bug 1232506: Make dom/devicestorage really work with e10s. r=alchen (7f95105c5e)
- Bug 1208944 - Part 1. Dispatch CompositionEvent to Plugin. r=masayuki (85c062b417)
- Bug 1208944 - Part 2-a. Handle CompositionEvent on plugin. r=masayuki (ea2cebfca9)
- Bug 1208944 - Part 2-b. Workaround for OSX. r=masayuki (ca401cbc04)
- Bug 1208944 - Part 3. Allow IME window messages on plugin process. r=jmathies (571fd75010)
- Bug 1192844: Accept 0xCC padding in WindowsDllDetourPatcher::CreateTrampoline. r=m_kato (b34b6173d3)
- Bug 1201205 part 1: Add an AutoVirtualProtect helper class to make the next patch easier. r=m_kato (b384bd2412)
- Bug 1201205 part 2: Restore protection on the nop space separately from the function. r=m_kato (a822b2414b)
- Bug 1208944 - Part 4. nsWindowsDllInterceptor supports IMM32 API hook. r=ehsan (9b409ff15a)
- Bug 1208944 - Part 5. Send PluginEvent to content process. r=jmathies (4ee0341190)
- Bug 1208944 - Part 6. Get vaild TextRangeArray on compositionupdate. r=masayuki (18f184931c)
- Bug 1208944 - Part 7. Don't post WM_IME_REQUEST on windowless plugin since we don't convert pointer over process. r=masayuki (6b1a9ce71f)
- Bug 1208944 - Part 8. Don't get selection on start compostion when plugin has foucs. r=masayuki (daf400620e)
- Add Telemetry for the drawing models that plugins use. (bug 1229961 part 1, r=aklotz,vladan) (c9645a68ec)
- Enable direct plugin drawing by default. (bug 1229961 part 2, r=aklotz) (080c9337cb)
- Bug 1213845 - enable osk support on windows 8, but hide it behind a preference, r=jaws (fcb8dfb10b)
- Bug 1192248 - Fix wchar_t/char16_t mismatch WinIMEHandler.cpp. r=masayuki (13e61f3f0c)
- Bug 1226145 - actually check whether the on-screen keyboard is up rather than relying on internal state, r=masayuki (da090b605d)
- Bug 1208944 - Part 9. Hook IMM32 APIs on plugin process. r=masayuki (f15d73721f)
- Bug 1208944 - Part 10-a. Call CallWindowProc when WidgetPluginEvent isn't handled by plugin. r=masayuki (c54eaa50ed)
- Bug 1208944 - Part 10-b. Call DefaultProc When CompositionEvent isn't handled correctly by plugin. r=masayuki (0e77eaa40f)
- Bug 1157046 - Remove ARRAY_LENGTH in favor of MOZ_ARRAY_LENGTH; r=Waldo (a7f8ce3684)
- Bug 1196834 - Add a test that confirms plugin windows are hidden after switching from a remote to local tab. r=roc (a09d4a32af)
- Bug 1208944 - Part 11. Add test. r=jmathies (ed8e4c87d2)
- Bug 1231378 - part 1 - Fix uninitialized members of classes in docshell/*, r=smaug (fcd7615d97)
- Bug 1231378 - part 2 - Fix uninitialized members of classes in dom/*,r=smaug (40b354438f)
- Bug 1231378 - part 3 - Fix uninitialized members of classes in module/libjar and mfbt, r=smaug (ee15d8739a)
- Bug 1231378 - part 4 - Fix uninitialized members of classes in netwerk/widget/storage/uriloader/memory/tools, r=smaug (6aec559dd8)
- Bug 1238082 - Fix mode lines in dom/ipc. r=baku (a49aa3555b)
- Bug 1024149 - Use element size for upload texture when using SVG image. r=jgilbert r=roc (531bdd2406)
- Bug 1233922 - Skip Camera preinit for browser elements. r=fabrice (cd8ffbb112)
- Bug 1159327 - Enable accessibility more broadly with e10s and add an e10s a11y blacklist for clients with known issues. r=tbsaunde (374f49a942)
- Bug 1198459 - Prevent accessibility from initializing in content processes when e10s is running. r=tbsaunde (33bdfccd56)
- Bug 1115956 - Improve notice string for when e10s was disabled for accessibility. r=mconley (e5ff276d9e)
- Bug 1172491 - Let e10s be enabled in safe mode. r=felipe (bd0a3fbbd5)
- Bug 1226487 - Allow e10s to run on Beta. r=mconley (05103c816d)
- Bug 634063 - Use gfxPrefs for some layers acceleration prefs. r=nical (9a20bcd8d2)
- Bug 1198459 - Add support for disabling e10s if a11y was run in the previous session, or run in a session over the previous week. r=felipe (7154a021bd)
- Bug 1171171 - Make sure the graphics preferences are initialized early enough. r=billm (34801701aa)
- Bug 1171171: Move nsAppRunner's gfxPrefs #include out of windows-specific section. r=milan a=KWierso (492fa00f99)
- Bug 1182048 - Part 1: Allow e10s to be enabled,r=vlad (628949ad83)
- Bug 1182048 - Part 2: Implement e10s support for WebVR,r=vlad (a77d280b48)
- Bug 1207221 - Do not prevent the system app from vibrating when it is hidden. r=bz r=dhylands (f3d2306f41)
- Bug 1233902 - Check the TP list in sendBeacon(). r=gcp (bc279755ab)
- Bug 1216207 - Modify navigator.hasFeature method to implement new (9b4571468e)
- Bug 1217187 - Modify navigator.hasFeature method to detect new (d9ea58e65c)
- Bug 1125477 - Part 1: Modify dom/tv mochitest in order to enable e10s test. r=seanlin (7b535fddaa)
- Bug 1125477 - Part 2: Modify permission check of TV Manager API in order to remove dom.testing.tv_enabled_for_hosted_apps. r=seanlin;r=smaug (5719fdf3a7)
- Bug 1125477 - Part3: Remove permission removing code. r=seanlin (614de5dfd1)
- Bug 1223672 - NSec package cannot use device storage API. r=kchen (f1fc6711ea)
- Bug 1224609 - Ensure the primary screen is always returned in RecvScreenForBrowser if all else fails. r=wmccloskey (d562935754)
- Bug 1180288 - Use native filepickers for Graphene. r=khuey (0b9c78ce13)
- Bug 1201394 - Use cached preferences value in ProcessPriorityManager. r=gsvelto (d211492014)
- Bug 1212833 - Delay the MemoryPressure when an application goes to background. r=gsvelto (ce33de4324)
- Bug 1186812 (part 1) - Replace nsBaseHashtable::EnumerateRead() calls in dom/{ipc,plugins}/. r=jimm. (c061bb9d1b)
- Bug 1212984 - HangMonitorChild should delete its Transport. r=billm (3331593b2d)
- Bug 1186812 (part 2) - Replace nsBaseHashtable::EnumerateRead() calls in dom/{ipc,plugins}/. r=jimm. (fdac3f52cb)
- Bug 1215239 - Do not set ownApp for a browser element. r=bholley (e3cc250c3f)
- Bug 1241278 - Change Notification.requestPermission() to return a promise. r=baku (b0a86eeeb5)
- Bug 1249102. Make overrides of WorkerRunnable::PostRun a bit more consistent. r=khuey (a23bf9468f)
- Bug 1209812 (part 6) - Convert all gfxImageFormat values to SurfaceFormat equivalents. r=jrmuizel. (9a5da3597f)
- Bug 1239225 - Remove unused args from TextureImage's constructor and related functions. r=mattwoodrow. (26ff005ecb)
- Bug 1240708 - Various trivial coverity warning fixes. r=kats (74a485c2b8)
- Bug 1240708 - Various trivial coverity warning fixes (part 2). r=kats (34bcd568dc)
- Bug 1219494 - Part 3 gfx/gl with gfxCrash. r=mchang (268968a3d9)
- Bug 1232456 - Create EGL surface through widget; r=snorp (13634bebc6)
- Bug 1232456 - Renew EGL surface using existing compositor widget; r=snorp (2649b088d3)
- Bug 1187322 - Don't require accelerated OpenGL contexts for BasicCompositor on OS X. r=jrmuizel (87f92af2c8)
- Bug 1238753 - Use StartRemoteDrawingInRegion on Mac. r=mattwoodrow (22870ed043)
- Bug 1238755 - Avoid a copy when uploading the BasicCompositor result to a texture. r=mattwoodrow (0b6663ab61)
- Bug 890156 - patch 0.2 - Remove the (unused) aRect parameter from nsBaseWidget::BaseCreate. r=kats (1120763eef)
- Bug 1241983 - Make DOM Identity observe inner-window-destroyed not dom-window-destroyed; r=ferjm (7cf4cf3211)
- Bug 1236282 - Clip color layer drawing in BasicCompositor so that unbounded operators don't erase stuff outside the layer. r=Bas (a7e73d1d8d)
- Bug 1238753 - Make BasicCompositor respect changes to mInvalidRegion through StartRemoteDrawingWithRegion properly. r=mattwoodrow (36e20bccce)
- Bug 1238753 - Don't skip the call to StartRemoteDrawing in from BasicCompositor::BeginFrame if the invalid region is empty. r=mattwoodrow (783b25775f)
- Bug 1239137 - Return early from BasicCompositor::DrawQuad if transformBounds is empty. r=mattwoodrow (0815cfc28a)
- Bug 1239530 (part 1) - Remove PuppetWidget::Scroll(), which is dead. r=kats. (63ab8c32b6)
- Bug 1239530 (part 2) - Use LayoutDevice coordinates in {Start,End}RemoteDrawingInRegion() and related functions. r=kats. (609001767d)
- Bug 1239537 - Remove Compositor::GetWidgetSize(), which is unused. r=mattwoodrow. (25732d37b1)
- Bug 1242293 - Don't call EndRemoteDrawingInRegion if StartRemoteDrawingInRegion returns a null DrawTarget. r=mstange (8e297c3b84)
- import changes from mozilla: Bug 1211642: Whitelist test plugin for async plugin init; r=jimm (f02deaaeb8) (6689ce99c8)
544 lines
18 KiB
C++
544 lines
18 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "ImageEncoder.h"
|
|
#include "mozilla/ClearOnShutdown.h"
|
|
#include "mozilla/dom/CanvasRenderingContext2D.h"
|
|
#include "mozilla/gfx/2D.h"
|
|
#include "mozilla/gfx/DataSurfaceHelpers.h"
|
|
#include "mozilla/layers/AsyncCanvasRenderer.h"
|
|
#include "mozilla/RefPtr.h"
|
|
#include "mozilla/SyncRunnable.h"
|
|
#include "mozilla/unused.h"
|
|
#include "gfxUtils.h"
|
|
#include "nsIThreadPool.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsXPCOMCIDInternal.h"
|
|
#include "WorkerPrivate.h"
|
|
|
|
using namespace mozilla::gfx;
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
// This class should be placed inside GetBRGADataSourceSurfaceSync(). However,
|
|
// due to B2G ICS uses old complier (C++98/03) which forbids local class as
|
|
// template parameter, we need to move this class outside.
|
|
class SurfaceHelper : public nsRunnable {
|
|
public:
|
|
explicit SurfaceHelper(already_AddRefed<layers::Image> aImage) : mImage(aImage) {}
|
|
|
|
// It retrieves a SourceSurface reference and convert color format on main
|
|
// thread and passes DataSourceSurface to caller thread.
|
|
NS_IMETHOD Run() {
|
|
// It guarantees the reference will be released on main thread.
|
|
nsCountedRef<nsMainThreadSourceSurfaceRef> surface;
|
|
surface.own(mImage->GetAsSourceSurface().take());
|
|
|
|
if (surface->GetFormat() == gfx::SurfaceFormat::B8G8R8A8) {
|
|
mDataSourceSurface = surface->GetDataSurface();
|
|
} else {
|
|
mDataSourceSurface = gfxUtils::
|
|
CopySurfaceToDataSourceSurfaceWithFormat(surface,
|
|
gfx::SurfaceFormat::B8G8R8A8);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<gfx::DataSourceSurface> GetDataSurfaceSafe() {
|
|
nsCOMPtr<nsIThread> mainThread = do_GetMainThread();
|
|
MOZ_ASSERT(mainThread);
|
|
SyncRunnable::DispatchToThread(mainThread, this, false);
|
|
|
|
return mDataSourceSurface.forget();
|
|
}
|
|
|
|
private:
|
|
RefPtr<layers::Image> mImage;
|
|
RefPtr<gfx::DataSourceSurface> mDataSourceSurface;
|
|
};
|
|
|
|
// This function returns a DataSourceSurface in B8G8R8A8 format.
|
|
// It uses SourceSurface to do format convert. Because most SourceSurface in
|
|
// image formats should be referenced or dereferenced on main thread, it uses a
|
|
// sync class SurfaceHelper to retrieve SourceSurface and convert to B8G8R8A8 on
|
|
// main thread.
|
|
already_AddRefed<DataSourceSurface>
|
|
GetBRGADataSourceSurfaceSync(already_AddRefed<layers::Image> aImage)
|
|
{
|
|
RefPtr<SurfaceHelper> helper = new SurfaceHelper(Move(aImage));
|
|
return helper->GetDataSurfaceSafe();
|
|
}
|
|
|
|
class EncodingCompleteEvent : public nsCancelableRunnable
|
|
{
|
|
virtual ~EncodingCompleteEvent() {}
|
|
|
|
public:
|
|
explicit EncodingCompleteEvent(EncodeCompleteCallback* aEncodeCompleteCallback)
|
|
: mImgSize(0)
|
|
, mType()
|
|
, mImgData(nullptr)
|
|
, mEncodeCompleteCallback(aEncodeCompleteCallback)
|
|
, mFailed(false)
|
|
{
|
|
if (!NS_IsMainThread() && workers::GetCurrentThreadWorkerPrivate()) {
|
|
mCreationThread = NS_GetCurrentThread();
|
|
} else {
|
|
NS_GetMainThread(getter_AddRefs(mCreationThread));
|
|
}
|
|
}
|
|
|
|
NS_IMETHOD Run() override
|
|
{
|
|
nsresult rv = NS_OK;
|
|
|
|
if (!mFailed) {
|
|
// The correct parentObject has to be set by the mEncodeCompleteCallback.
|
|
RefPtr<Blob> blob =
|
|
Blob::CreateMemoryBlob(nullptr, mImgData, mImgSize, mType);
|
|
MOZ_ASSERT(blob);
|
|
|
|
rv = mEncodeCompleteCallback->ReceiveBlob(blob.forget());
|
|
}
|
|
|
|
mEncodeCompleteCallback = nullptr;
|
|
|
|
return rv;
|
|
}
|
|
|
|
void SetMembers(void* aImgData, uint64_t aImgSize, const nsAutoString& aType)
|
|
{
|
|
mImgData = aImgData;
|
|
mImgSize = aImgSize;
|
|
mType = aType;
|
|
}
|
|
|
|
void SetFailed()
|
|
{
|
|
mFailed = true;
|
|
}
|
|
|
|
nsIThread* GetCreationThread()
|
|
{
|
|
return mCreationThread;
|
|
}
|
|
|
|
private:
|
|
uint64_t mImgSize;
|
|
nsAutoString mType;
|
|
void* mImgData;
|
|
nsCOMPtr<nsIThread> mCreationThread;
|
|
RefPtr<EncodeCompleteCallback> mEncodeCompleteCallback;
|
|
bool mFailed;
|
|
};
|
|
|
|
class EncodingRunnable : public nsRunnable
|
|
{
|
|
virtual ~EncodingRunnable() {}
|
|
|
|
public:
|
|
NS_DECL_ISUPPORTS_INHERITED
|
|
|
|
EncodingRunnable(const nsAString& aType,
|
|
const nsAString& aOptions,
|
|
UniquePtr<uint8_t[]> aImageBuffer,
|
|
layers::Image* aImage,
|
|
imgIEncoder* aEncoder,
|
|
EncodingCompleteEvent* aEncodingCompleteEvent,
|
|
int32_t aFormat,
|
|
const nsIntSize aSize,
|
|
bool aUsingCustomOptions)
|
|
: mType(aType)
|
|
, mOptions(aOptions)
|
|
, mImageBuffer(Move(aImageBuffer))
|
|
, mImage(aImage)
|
|
, mEncoder(aEncoder)
|
|
, mEncodingCompleteEvent(aEncodingCompleteEvent)
|
|
, mFormat(aFormat)
|
|
, mSize(aSize)
|
|
, mUsingCustomOptions(aUsingCustomOptions)
|
|
{}
|
|
|
|
nsresult ProcessImageData(uint64_t* aImgSize, void** aImgData)
|
|
{
|
|
nsCOMPtr<nsIInputStream> stream;
|
|
nsresult rv = ImageEncoder::ExtractDataInternal(mType,
|
|
mOptions,
|
|
mImageBuffer.get(),
|
|
mFormat,
|
|
mSize,
|
|
mImage,
|
|
nullptr,
|
|
nullptr,
|
|
getter_AddRefs(stream),
|
|
mEncoder);
|
|
|
|
// If there are unrecognized custom parse options, we should fall back to
|
|
// the default values for the encoder without any options at all.
|
|
if (rv == NS_ERROR_INVALID_ARG && mUsingCustomOptions) {
|
|
rv = ImageEncoder::ExtractDataInternal(mType,
|
|
EmptyString(),
|
|
mImageBuffer.get(),
|
|
mFormat,
|
|
mSize,
|
|
mImage,
|
|
nullptr,
|
|
nullptr,
|
|
getter_AddRefs(stream),
|
|
mEncoder);
|
|
}
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
rv = stream->Available(aImgSize);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
NS_ENSURE_TRUE(*aImgSize <= UINT32_MAX, NS_ERROR_FILE_TOO_BIG);
|
|
|
|
rv = NS_ReadInputStreamToBuffer(stream, aImgData, *aImgSize);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHOD Run() override
|
|
{
|
|
uint64_t imgSize;
|
|
void* imgData = nullptr;
|
|
|
|
nsresult rv = ProcessImageData(&imgSize, &imgData);
|
|
if (NS_FAILED(rv)) {
|
|
mEncodingCompleteEvent->SetFailed();
|
|
} else {
|
|
mEncodingCompleteEvent->SetMembers(imgData, imgSize, mType);
|
|
}
|
|
rv = mEncodingCompleteEvent->GetCreationThread()->
|
|
Dispatch(mEncodingCompleteEvent, nsIThread::DISPATCH_NORMAL);
|
|
if (NS_FAILED(rv)) {
|
|
// Better to leak than to crash.
|
|
Unused << mEncodingCompleteEvent.forget();
|
|
return rv;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
private:
|
|
nsAutoString mType;
|
|
nsAutoString mOptions;
|
|
UniquePtr<uint8_t[]> mImageBuffer;
|
|
RefPtr<layers::Image> mImage;
|
|
nsCOMPtr<imgIEncoder> mEncoder;
|
|
RefPtr<EncodingCompleteEvent> mEncodingCompleteEvent;
|
|
int32_t mFormat;
|
|
const nsIntSize mSize;
|
|
bool mUsingCustomOptions;
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS_INHERITED0(EncodingRunnable, nsRunnable);
|
|
|
|
StaticRefPtr<nsIThreadPool> ImageEncoder::sThreadPool;
|
|
|
|
/* static */
|
|
nsresult
|
|
ImageEncoder::ExtractData(nsAString& aType,
|
|
const nsAString& aOptions,
|
|
const nsIntSize aSize,
|
|
nsICanvasRenderingContextInternal* aContext,
|
|
layers::AsyncCanvasRenderer* aRenderer,
|
|
nsIInputStream** aStream)
|
|
{
|
|
nsCOMPtr<imgIEncoder> encoder = ImageEncoder::GetImageEncoder(aType);
|
|
if (!encoder) {
|
|
return NS_IMAGELIB_ERROR_NO_ENCODER;
|
|
}
|
|
|
|
return ExtractDataInternal(aType, aOptions, nullptr, 0, aSize, nullptr,
|
|
aContext, aRenderer, aStream, encoder);
|
|
}
|
|
|
|
/* static */
|
|
nsresult
|
|
ImageEncoder::ExtractDataFromLayersImageAsync(nsAString& aType,
|
|
const nsAString& aOptions,
|
|
bool aUsingCustomOptions,
|
|
layers::Image* aImage,
|
|
EncodeCompleteCallback* aEncodeCallback)
|
|
{
|
|
nsCOMPtr<imgIEncoder> encoder = ImageEncoder::GetImageEncoder(aType);
|
|
if (!encoder) {
|
|
return NS_IMAGELIB_ERROR_NO_ENCODER;
|
|
}
|
|
|
|
nsresult rv = EnsureThreadPool();
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
RefPtr<EncodingCompleteEvent> completeEvent =
|
|
new EncodingCompleteEvent(aEncodeCallback);
|
|
|
|
nsIntSize size(aImage->GetSize().width, aImage->GetSize().height);
|
|
nsCOMPtr<nsIRunnable> event = new EncodingRunnable(aType,
|
|
aOptions,
|
|
nullptr,
|
|
aImage,
|
|
encoder,
|
|
completeEvent,
|
|
imgIEncoder::INPUT_FORMAT_HOSTARGB,
|
|
size,
|
|
aUsingCustomOptions);
|
|
return sThreadPool->Dispatch(event, NS_DISPATCH_NORMAL);
|
|
}
|
|
|
|
/* static */
|
|
nsresult
|
|
ImageEncoder::ExtractDataAsync(nsAString& aType,
|
|
const nsAString& aOptions,
|
|
bool aUsingCustomOptions,
|
|
UniquePtr<uint8_t[]> aImageBuffer,
|
|
int32_t aFormat,
|
|
const nsIntSize aSize,
|
|
EncodeCompleteCallback* aEncodeCallback)
|
|
{
|
|
nsCOMPtr<imgIEncoder> encoder = ImageEncoder::GetImageEncoder(aType);
|
|
if (!encoder) {
|
|
return NS_IMAGELIB_ERROR_NO_ENCODER;
|
|
}
|
|
|
|
nsresult rv = EnsureThreadPool();
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
RefPtr<EncodingCompleteEvent> completeEvent =
|
|
new EncodingCompleteEvent(aEncodeCallback);
|
|
|
|
nsCOMPtr<nsIRunnable> event = new EncodingRunnable(aType,
|
|
aOptions,
|
|
Move(aImageBuffer),
|
|
nullptr,
|
|
encoder,
|
|
completeEvent,
|
|
aFormat,
|
|
aSize,
|
|
aUsingCustomOptions);
|
|
return sThreadPool->Dispatch(event, NS_DISPATCH_NORMAL);
|
|
}
|
|
|
|
/*static*/ nsresult
|
|
ImageEncoder::GetInputStream(int32_t aWidth,
|
|
int32_t aHeight,
|
|
uint8_t* aImageBuffer,
|
|
int32_t aFormat,
|
|
imgIEncoder* aEncoder,
|
|
const char16_t* aEncoderOptions,
|
|
nsIInputStream** aStream)
|
|
{
|
|
nsresult rv =
|
|
aEncoder->InitFromData(aImageBuffer,
|
|
aWidth * aHeight * 4, aWidth, aHeight, aWidth * 4,
|
|
aFormat,
|
|
nsDependentString(aEncoderOptions));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
return CallQueryInterface(aEncoder, aStream);
|
|
}
|
|
|
|
/* static */
|
|
nsresult
|
|
ImageEncoder::ExtractDataInternal(const nsAString& aType,
|
|
const nsAString& aOptions,
|
|
uint8_t* aImageBuffer,
|
|
int32_t aFormat,
|
|
const nsIntSize aSize,
|
|
layers::Image* aImage,
|
|
nsICanvasRenderingContextInternal* aContext,
|
|
layers::AsyncCanvasRenderer* aRenderer,
|
|
nsIInputStream** aStream,
|
|
imgIEncoder* aEncoder)
|
|
{
|
|
if (aSize.IsEmpty()) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
nsCOMPtr<nsIInputStream> imgStream;
|
|
|
|
// get image bytes
|
|
nsresult rv;
|
|
if (aImageBuffer) {
|
|
rv = ImageEncoder::GetInputStream(
|
|
aSize.width,
|
|
aSize.height,
|
|
aImageBuffer,
|
|
aFormat,
|
|
aEncoder,
|
|
nsPromiseFlatString(aOptions).get(),
|
|
getter_AddRefs(imgStream));
|
|
} else if (aContext) {
|
|
NS_ConvertUTF16toUTF8 encoderType(aType);
|
|
rv = aContext->GetInputStream(encoderType.get(),
|
|
nsPromiseFlatString(aOptions).get(),
|
|
getter_AddRefs(imgStream));
|
|
} else if (aRenderer) {
|
|
NS_ConvertUTF16toUTF8 encoderType(aType);
|
|
rv = aRenderer->GetInputStream(encoderType.get(),
|
|
nsPromiseFlatString(aOptions).get(),
|
|
getter_AddRefs(imgStream));
|
|
} else if (aImage) {
|
|
// It is safe to convert PlanarYCbCr format from YUV to RGB off-main-thread.
|
|
// Other image formats could have problem to convert format off-main-thread.
|
|
// So here it uses a help function GetBRGADataSourceSurfaceSync() to convert
|
|
// format on main thread.
|
|
if (aImage->GetFormat() == ImageFormat::PLANAR_YCBCR) {
|
|
nsTArray<uint8_t> data;
|
|
layers::PlanarYCbCrImage* ycbcrImage = static_cast<layers::PlanarYCbCrImage*> (aImage);
|
|
gfxImageFormat format = SurfaceFormat::A8R8G8B8_UINT32;
|
|
uint32_t stride = GetAlignedStride<16>(aSize.width * 4);
|
|
size_t length = BufferSizeFromStrideAndHeight(stride, aSize.height);
|
|
data.SetCapacity(length);
|
|
|
|
gfxUtils::ConvertYCbCrToRGB(*ycbcrImage->GetData(),
|
|
format,
|
|
aSize,
|
|
data.Elements(),
|
|
stride);
|
|
|
|
rv = aEncoder->InitFromData(data.Elements(),
|
|
aSize.width * aSize.height * 4,
|
|
aSize.width,
|
|
aSize.height,
|
|
aSize.width * 4,
|
|
imgIEncoder::INPUT_FORMAT_HOSTARGB,
|
|
aOptions);
|
|
} else {
|
|
RefPtr<gfx::DataSourceSurface> dataSurface;
|
|
RefPtr<layers::Image> image(aImage);
|
|
dataSurface = GetBRGADataSourceSurfaceSync(image.forget());
|
|
|
|
DataSourceSurface::MappedSurface map;
|
|
if (!dataSurface->Map(gfx::DataSourceSurface::MapType::READ, &map)) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
rv = aEncoder->InitFromData(map.mData,
|
|
aSize.width * aSize.height * 4,
|
|
aSize.width,
|
|
aSize.height,
|
|
aSize.width * 4,
|
|
imgIEncoder::INPUT_FORMAT_HOSTARGB,
|
|
aOptions);
|
|
dataSurface->Unmap();
|
|
}
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
imgStream = do_QueryInterface(aEncoder);
|
|
}
|
|
} else {
|
|
CheckedInt32 requiredBytes = CheckedInt32(aSize.width) * CheckedInt32(aSize.height) * 4;
|
|
if (MOZ_UNLIKELY(!requiredBytes.isValid())) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
// no context, so we have to encode an empty image
|
|
// note that if we didn't have a current context, the spec says we're
|
|
// supposed to just return transparent black pixels of the canvas
|
|
// dimensions.
|
|
RefPtr<DataSourceSurface> emptyCanvas =
|
|
Factory::CreateDataSourceSurfaceWithStride(IntSize(aSize.width, aSize.height),
|
|
SurfaceFormat::B8G8R8A8,
|
|
4 * aSize.width, true);
|
|
if (NS_WARN_IF(!emptyCanvas)) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
DataSourceSurface::MappedSurface map;
|
|
if (!emptyCanvas->Map(DataSourceSurface::MapType::WRITE, &map)) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
rv = aEncoder->InitFromData(map.mData,
|
|
aSize.width * aSize.height * 4,
|
|
aSize.width,
|
|
aSize.height,
|
|
aSize.width * 4,
|
|
imgIEncoder::INPUT_FORMAT_HOSTARGB,
|
|
aOptions);
|
|
emptyCanvas->Unmap();
|
|
if (NS_SUCCEEDED(rv)) {
|
|
imgStream = do_QueryInterface(aEncoder);
|
|
}
|
|
}
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
imgStream.forget(aStream);
|
|
return rv;
|
|
}
|
|
|
|
/* static */
|
|
already_AddRefed<imgIEncoder>
|
|
ImageEncoder::GetImageEncoder(nsAString& aType)
|
|
{
|
|
// Get an image encoder for the media type.
|
|
nsCString encoderCID("@mozilla.org/image/encoder;2?type=");
|
|
NS_ConvertUTF16toUTF8 encoderType(aType);
|
|
encoderCID += encoderType;
|
|
nsCOMPtr<imgIEncoder> encoder = do_CreateInstance(encoderCID.get());
|
|
|
|
if (!encoder && aType != NS_LITERAL_STRING("image/png")) {
|
|
// Unable to create an encoder instance of the specified type. Falling back
|
|
// to PNG.
|
|
aType.AssignLiteral("image/png");
|
|
nsCString PNGEncoderCID("@mozilla.org/image/encoder;2?type=image/png");
|
|
encoder = do_CreateInstance(PNGEncoderCID.get());
|
|
}
|
|
|
|
return encoder.forget();
|
|
}
|
|
|
|
/* static */
|
|
nsresult
|
|
ImageEncoder::EnsureThreadPool()
|
|
{
|
|
if (!sThreadPool) {
|
|
nsCOMPtr<nsIThreadPool> threadPool = do_CreateInstance(NS_THREADPOOL_CONTRACTID);
|
|
sThreadPool = threadPool;
|
|
if (!NS_IsMainThread()) {
|
|
NS_DispatchToMainThread(NS_NewRunnableFunction([]() -> void {
|
|
ClearOnShutdown(&sThreadPool);
|
|
}));
|
|
} else {
|
|
ClearOnShutdown(&sThreadPool);
|
|
}
|
|
|
|
const uint32_t kThreadLimit = 2;
|
|
const uint32_t kIdleThreadLimit = 1;
|
|
const uint32_t kIdleThreadTimeoutMs = 30000;
|
|
|
|
nsresult rv = sThreadPool->SetName(NS_LITERAL_CSTRING("EncodingRunnable"));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = sThreadPool->SetThreadLimit(kThreadLimit);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = sThreadPool->SetIdleThreadLimit(kIdleThreadLimit);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = sThreadPool->SetIdleThreadTimeout(kIdleThreadTimeoutMs);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|