mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 05:37:11 +00:00
8f529f64f5
- Bug 932865 - Add ThreadHangStats for collecting background hang telemetry; r=vladan (2f08a076b)
- Bug 932865 - Add way for telemetry to iterate over active threads; r=froydnj (535615d3d)
- Bug 1128768: Part 3 - Update BHR to allow for hang annotations; r=vladan (0b880a667)
- Bug 935092 - Add ThreadStackHelper to get a thread's pesudo-stack; r=BenWa (1422cfe4d)
- Bug 942488 - Don't report pseudo-stacks without SPS profiler. r=nchen (e160a7a08)
- Bug 946817 - Don't assert mPseudoStack on B2G. r=BenWa (9f846df3b)
- Bug 951431 - Don't get stacks during profiler runs on Linux; r=BenWa (15036e907)
- Bug 978262 - Ignore duplicate frames when getting BHR stack. r=froydnj (964721b1b)
- Bug 985155 - Add signal trampoline on ARM Linux to work around kernel bug. r=snorp (cb8a7846c)
- Bug 995730 - Convert xpcom/threads/ to Gecko style. r=froydnj (fe150404e)
- Bug 1013326 - Distinguish chrome and content scripts in pseudostack; r=snorp (81273c977)
- Bug 1023461 - Remove temporary stack buffer in ThreadStackHelper; r=snorp (cf5a717c2)
- Bug 1023461 - Record filename and line number for chrome JS entries; r=snorp (10c89808f)
- Bug 1022456 - Fix modelines in xpcom/{base,glue,io,string,threads}/. (48dbc0416)
- Bug 1016441 - Switch to using real-time signal in ThreadStackHelper; (2c5f818be)
- Bug 1016629 - b. Use RAII class to assign mStackToFill; r=snorp (769eae130)
- Bug 1016629 - c. Add define for ThreadStackHelper pseudostack support; r=snorp (67def0d2f)
- Bug 1016629 - d. Add and implement GetNativeStack method in ThreadStackHelper; r=snorp r=jseward (46c52f2be)
- Bug 1016629 - e. Implement platform-specific code for filling in context; r=snorp r=jseward (e6a66858b)
- Bug 1016629 - g. Avoid ASan flag when copying stack; r=snorp (0159628b5)
- Bug 1045176 - Unbreak build on non-SPS platforms after bug 1016629. (f1d60d838)
- Bug 1047123 - ThreadStackHelper should use UniquePtr<uint8_t[]>, not ScopedDeleteArray. r=jchen (0e4af313c)
- Bug 1049161 - Fix ThreadStackHelper thread handle permissions on Windows; r=snorp (c05172b1c)
- Bug 1050185 - Make ThreadStackHelper::FillThreadContext Valgrind-friendly. r=nchen (368725774)
- Bug 1050440 - Remove repeated js::RunScript frames in ThreadStackHelper (2a79600b3)
- Bug 1046841 - Fix more style violations in previously touched .cpp files in xpcom/. r=froydnj (02afe2493)
- Bug 1069694 - Remove or move around functions in OldDebugAPI. r=shu (177197302)
- Bug 1069694 - Remove OldDebugAPI from the browser. r=shu (b8c917d42)
- Bug 1100911 - For MacOS builds running on Valgrind, make ThreadStackHelper::GetStack be a no-op. r=nchen. (d99c02e16)
- Bug 1091758 - Report full paths for most chrome scripts; r=snorp (2b72e7878)
- Bug 1109291 - Include better paths for hanging chrome scripts in profile extensions directory; r=snorp r=bsmedberg (1997b9532)
- Bug 1113416 - Don't read stack labels inside hang monitor sighandler; r=nfroyd r=snorp (9688f6069)
- bug 1146027 - more final r=froydnj (7b0f295e5)
- Bug 1164090 - Check for Windows path separator in BHR file name; r=snorp (f014b4d78)
- Bug 1169034 - include <cstdlib> in ThreadStackHelper.cpp to declare correct overload for std::abs; r=jseward (874d4447e)
- Bug 1182996 - Fix and add missing namespace comments. rs=ehsan (054fc00b2)
- Bug 932865 - Collect thread hang stats in BackgroundHangMonitor; (ac80c8e9f)
- minor anticipated fixes to get it compiling (2bd701d15)
252 lines
7.4 KiB
C++
252 lines
7.4 KiB
C++
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
* 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 "ImageScaling.h"
|
|
#include "2D.h"
|
|
#include "DataSurfaceHelpers.h"
|
|
|
|
#include <math.h>
|
|
#include <algorithm>
|
|
|
|
using namespace std;
|
|
|
|
namespace mozilla {
|
|
namespace gfx {
|
|
|
|
inline uint32_t Avg2x2(uint32_t a, uint32_t b, uint32_t c, uint32_t d)
|
|
{
|
|
// Prepare half-adder work
|
|
uint32_t sum = a ^ b ^ c;
|
|
uint32_t carry = (a & b) | (a & c) | (b & c);
|
|
|
|
// Before shifting, mask lower order bits of each byte to avoid underflow.
|
|
uint32_t mask = 0xfefefefe;
|
|
|
|
// Add d to sum and divide by 2.
|
|
sum = (((sum ^ d) & mask) >> 1) + (sum & d);
|
|
|
|
// Sum is now shifted into place relative to carry, add them together.
|
|
return (((sum ^ carry) & mask) >> 1) + (sum & carry);
|
|
}
|
|
|
|
inline uint32_t Avg2(uint32_t a, uint32_t b)
|
|
{
|
|
// Prepare half-adder work
|
|
uint32_t sum = a ^ b;
|
|
uint32_t carry = (a & b);
|
|
|
|
// Before shifting, mask lower order bits of each byte to avoid underflow.
|
|
uint32_t mask = 0xfefefefe;
|
|
|
|
// Add d to sum and divide by 2.
|
|
return ((sum & mask) >> 1) + carry;
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::ScaleForSize(const IntSize &aSize)
|
|
{
|
|
uint32_t horizontalDownscales = 0;
|
|
uint32_t verticalDownscales = 0;
|
|
|
|
IntSize scaleSize = mOrigSize;
|
|
while ((scaleSize.height / 2) > aSize.height) {
|
|
verticalDownscales++;
|
|
scaleSize.height /= 2;
|
|
}
|
|
|
|
while ((scaleSize.width / 2) > aSize.width) {
|
|
horizontalDownscales++;
|
|
scaleSize.width /= 2;
|
|
}
|
|
|
|
if (scaleSize == mOrigSize) {
|
|
return;
|
|
}
|
|
|
|
IntSize internalSurfSize;
|
|
|
|
internalSurfSize.width = max(scaleSize.width, mOrigSize.width / 2);
|
|
internalSurfSize.height = max(scaleSize.height, mOrigSize.height / 2);
|
|
|
|
mStride = internalSurfSize.width * 4;
|
|
if (mStride % 16) {
|
|
mStride += 16 - (mStride % 16);
|
|
}
|
|
|
|
delete [] mDataStorage;
|
|
// Allocate 15 bytes extra to make sure we can get 16 byte alignment. We
|
|
// should add tools for this, see bug 751696.
|
|
size_t bufLen = BufferSizeFromStrideAndHeight(mStride, internalSurfSize.height, 15);
|
|
if (bufLen == 0) {
|
|
mSize.SizeTo(0, 0);
|
|
mDataStorage = nullptr;
|
|
return;
|
|
}
|
|
mDataStorage = new uint8_t[bufLen];
|
|
|
|
if (uintptr_t(mDataStorage) % 16) {
|
|
// Our storage does not start at a 16-byte boundary. Make sure mData does!
|
|
mData = (uint8_t*)(uintptr_t(mDataStorage) +
|
|
(16 - (uintptr_t(mDataStorage) % 16)));
|
|
} else {
|
|
mData = mDataStorage;
|
|
}
|
|
|
|
mSize = scaleSize;
|
|
|
|
/* The surface we sample from might not be even sized, if it's not we will
|
|
* ignore the last row/column. This means we lose some data but it keeps the
|
|
* code very simple. There's also no perfect answer that provides a better
|
|
* solution.
|
|
*/
|
|
IntSize currentSampledSize = mOrigSize;
|
|
uint32_t currentSampledStride = mOrigStride;
|
|
uint8_t *currentSampledData = mOrigData;
|
|
|
|
while (verticalDownscales && horizontalDownscales) {
|
|
if (currentSampledSize.width % 2) {
|
|
currentSampledSize.width -= 1;
|
|
}
|
|
if (currentSampledSize.height % 2) {
|
|
currentSampledSize.height -= 1;
|
|
}
|
|
|
|
HalfImage2D(currentSampledData, currentSampledStride, currentSampledSize,
|
|
mData, mStride);
|
|
|
|
verticalDownscales--;
|
|
horizontalDownscales--;
|
|
currentSampledSize.width /= 2;
|
|
currentSampledSize.height /= 2;
|
|
currentSampledData = mData;
|
|
currentSampledStride = mStride;
|
|
}
|
|
|
|
while (verticalDownscales) {
|
|
if (currentSampledSize.height % 2) {
|
|
currentSampledSize.height -= 1;
|
|
}
|
|
|
|
HalfImageVertical(currentSampledData, currentSampledStride, currentSampledSize,
|
|
mData, mStride);
|
|
|
|
verticalDownscales--;
|
|
currentSampledSize.height /= 2;
|
|
currentSampledData = mData;
|
|
currentSampledStride = mStride;
|
|
}
|
|
|
|
|
|
while (horizontalDownscales) {
|
|
if (currentSampledSize.width % 2) {
|
|
currentSampledSize.width -= 1;
|
|
}
|
|
|
|
HalfImageHorizontal(currentSampledData, currentSampledStride, currentSampledSize,
|
|
mData, mStride);
|
|
|
|
horizontalDownscales--;
|
|
currentSampledSize.width /= 2;
|
|
currentSampledData = mData;
|
|
currentSampledStride = mStride;
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::HalfImage2D(uint8_t *aSource, int32_t aSourceStride,
|
|
const IntSize &aSourceSize, uint8_t *aDest,
|
|
uint32_t aDestStride)
|
|
{
|
|
#ifdef USE_SSE2
|
|
if (Factory::HasSSE2()) {
|
|
HalfImage2D_SSE2(aSource, aSourceStride, aSourceSize, aDest, aDestStride);
|
|
} else
|
|
#endif
|
|
{
|
|
HalfImage2D_C(aSource, aSourceStride, aSourceSize, aDest, aDestStride);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::HalfImageVertical(uint8_t *aSource, int32_t aSourceStride,
|
|
const IntSize &aSourceSize, uint8_t *aDest,
|
|
uint32_t aDestStride)
|
|
{
|
|
#ifdef USE_SSE2
|
|
if (Factory::HasSSE2()) {
|
|
HalfImageVertical_SSE2(aSource, aSourceStride, aSourceSize, aDest, aDestStride);
|
|
} else
|
|
#endif
|
|
{
|
|
HalfImageVertical_C(aSource, aSourceStride, aSourceSize, aDest, aDestStride);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::HalfImageHorizontal(uint8_t *aSource, int32_t aSourceStride,
|
|
const IntSize &aSourceSize, uint8_t *aDest,
|
|
uint32_t aDestStride)
|
|
{
|
|
#ifdef USE_SSE2
|
|
if (Factory::HasSSE2()) {
|
|
HalfImageHorizontal_SSE2(aSource, aSourceStride, aSourceSize, aDest, aDestStride);
|
|
} else
|
|
#endif
|
|
{
|
|
HalfImageHorizontal_C(aSource, aSourceStride, aSourceSize, aDest, aDestStride);
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::HalfImage2D_C(uint8_t *aSource, int32_t aSourceStride,
|
|
const IntSize &aSourceSize, uint8_t *aDest,
|
|
uint32_t aDestStride)
|
|
{
|
|
for (int y = 0; y < aSourceSize.height; y += 2) {
|
|
uint32_t *storage = (uint32_t*)(aDest + (y / 2) * aDestStride);
|
|
for (int x = 0; x < aSourceSize.width; x += 2) {
|
|
uint8_t *upperRow = aSource + (y * aSourceStride + x * 4);
|
|
uint8_t *lowerRow = aSource + ((y + 1) * aSourceStride + x * 4);
|
|
|
|
*storage++ = Avg2x2(*(uint32_t*)upperRow, *((uint32_t*)upperRow + 1),
|
|
*(uint32_t*)lowerRow, *((uint32_t*)lowerRow + 1));
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::HalfImageVertical_C(uint8_t *aSource, int32_t aSourceStride,
|
|
const IntSize &aSourceSize, uint8_t *aDest,
|
|
uint32_t aDestStride)
|
|
{
|
|
for (int y = 0; y < aSourceSize.height; y += 2) {
|
|
uint32_t *storage = (uint32_t*)(aDest + (y / 2) * aDestStride);
|
|
for (int x = 0; x < aSourceSize.width; x++) {
|
|
uint32_t *upperRow = (uint32_t*)(aSource + (y * aSourceStride + x * 4));
|
|
uint32_t *lowerRow = (uint32_t*)(aSource + ((y + 1) * aSourceStride + x * 4));
|
|
|
|
*storage++ = Avg2(*upperRow, *lowerRow);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
ImageHalfScaler::HalfImageHorizontal_C(uint8_t *aSource, int32_t aSourceStride,
|
|
const IntSize &aSourceSize, uint8_t *aDest,
|
|
uint32_t aDestStride)
|
|
{
|
|
for (int y = 0; y < aSourceSize.height; y++) {
|
|
uint32_t *storage = (uint32_t*)(aDest + y * aDestStride);
|
|
for (int x = 0; x < aSourceSize.width; x+= 2) {
|
|
uint32_t *pixels = (uint32_t*)(aSource + (y * aSourceStride + x * 4));
|
|
|
|
*storage++ = Avg2(*pixels, *(pixels + 1));
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace gfx
|
|
} // namespace mozilla
|