Hi,
I've still been tracking memory leaks in our application, and I think I have a
good test scenario to share.
In our application we found that a lot of memory would be held onto when
visiting this particular site:
http://answers.ea.com
In a loop we visit this site, and a simple html page that consists of a list of
test urls found on the local hard drive.
Our in built memory tools suggest a leak is coming largely from
CSSParser::parseFontFaceSrc(), though admittedly I've changed my top suspects
from time to time. I think this item is showing as a large leak due to the
inclusion of the font in the css on the answers.ea.com page, but perhaps there
are more smaller items being leaked which is just less obvious (the font is
large).
I've reproduce the scenario in the WinCairo test application, based on 188436.
I built in a quick and hacky soak mechanism to conduct the test, perhaps there
are better tools I'm not aware of. I've attached the source changes to this
mail:
Source/WebKit/win/WebKitMessageLoop.cpp
Tools/WinLauncher/Common.cpp
I can see the memory of the WinLauncher process grow over time, here is an
example of the memory usage as seen from the windows task manager over time:
Duration Memory (reported by windows)
0m 155MB
13m 178MB
23m 199MB
36m 219MB
53m 247MB
1h4m 259MB
1h23m 277MB
After about an 1h30m, WinLauncher crashes, at the end of the mail is the basic
debug data that was available.
I'm not sure if the crash is related to the memory leak I'm experiencing, or if
it's an unrelated issue. Since we're a few months older from tip I'm wondering
if anyone has any recent insights to share on this memory leak scenario? I
haven't identified anything in the change logs, but perhaps there is there
something already addressing this? Does the same issue occur at tip?
Any feedback is appreciated,
Thanks
Chris Vienneau
+ this 0x00000000784be130
{m_fontMetrics={m_unitsPerEm=1000 m_ascent=0.000000000 m_descent=0.000000000
...} ...} WebCore::Font *
ascent 0.000000000 float
dc Variable is optimized away and not
available.
+ extents {x_bearing=0.00000000000000000
y_bearing=2.840573810841e-316#DEN width=1.197610077640e-309#DEN ...}
cairo_text_extents_t
metricsMultiplier
0.031250000000000000 const double
faceName Variable is optimized away
and not available.
descent 52852716.0 float
xHeight Variable is optimized away and not
available.
+ textMetrics {tmHeight=0 tmAscent=0
tmDescent=1691286912 ...} tagTEXTMETRICW
lineGap 63069964.0 float
scaledFont 0x000000006477fc00 {...}
_cairo_scaled_font *
faceLength 0 int
> WebKit.dll!WebCore::Font::platformInit() Line 82 C++
WebKit.dll!WebCore::Font::Font(const WebCore::FontPlatformData &
platformData, bool isCustomFont, bool isLoading, bool
isTextOrientationFallback) Line 80 C++
WebKit.dll!WebCore::CachedFont::createFont(const
WebCore::FontDescription & fontDescription, const WTF::AtomicString & __formal,
bool syntheticBold, bool syntheticItalic, bool __formal) Line 126
C++
WebKit.dll!WebCore::CSSFontFaceSource::font(const
WebCore::FontDescription & fontDescription, bool syntheticBold, bool
syntheticItalic, WebCore::CSSFontSelector * fontSelector) Line 138 C++
WebKit.dll!WebCore::CSSFontFace::font(const
WebCore::FontDescription & fontDescription, bool syntheticBold, bool
syntheticItalic) Line 127 C++
WebKit.dll!WebCore::CSSSegmentedFontFace::fontRanges(const
WebCore::FontDescription & fontDescription) Line 130 C++
WebKit.dll!WebCore::CSSFontSelector::fontRangesForFamily(const
WebCore::FontDescription & fontDescription, const WTF::AtomicString &
familyName) Line 474 C++
WebKit.dll!WebCore::realizeNextFallback(const
WebCore::FontDescription & description, unsigned int & index,
WebCore::FontSelector * fontSelector) Line 90 C++
WebKit.dll!WebCore::FontCascadeFonts::realizeFallbackRangesAt(const
WebCore::FontDescription & description, unsigned int index) Line 114
C++
WebKit.dll!WebCore::FontCascadeFonts::determinePitch(const
WebCore::FontDescription & description) Line 64 C++
WebKit.dll!WebCore::BreakingContext::handleText(WTF::Vector<WebCore::WordMeasurement,64,WTF::CrashOnOverflow,16>
& wordMeasurements, bool & hyphenated, unsigned int &
consecutiveHyphenatedLines) Line 731 C++
WebKit.dll!WebCore::LineBreaker::nextLineBreak(WebCore::BidiResolver<WebCore::InlineIterator,WebCore::BidiRun>
& resolver, WebCore::LineInfo & lineInfo, WebCore::RenderTextInfo &
renderTextInfo, WebCore::FloatingObject * lastFloatFromPreviousLine, unsigned
int consecutiveHyphenatedLines,
WTF::Vector<WebCore::WordMeasurement,64,WTF::CrashOnOverflow,16> &
wordMeasurements) Line 110 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutRunsAndFloatsInRange(WebCore::LineLayoutState
& layoutState, WebCore::BidiResolver<WebCore::InlineIterator,WebCore::BidiRun>
& resolver, const WebCore::InlineIterator & cleanLineStart, const
WebCore::BidiStatus & cleanLineBidiStatus, unsigned int
consecutiveHyphenatedLines) Line 1260 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutRunsAndFloats(WebCore::LineLayoutState
& layoutState, bool hasInlineChild) Line 1214 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutLineBoxes(bool
relayoutChildren, WebCore::LayoutUnit & repaintLogicalTop, WebCore::LayoutUnit
& repaintLogicalBottom) Line 1624 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 484 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::insertFloatingObject(WebCore::RenderBox &
floatBox) Line 2247 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 627
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChild(WebCore::RenderBox &
child, WebCore::RenderBlockFlow::MarginInfo & marginInfo, WebCore::LayoutUnit &
previousFloatLogicalBottom, WebCore::LayoutUnit & maxFloatLogicalBottom) Line
712 C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlockChildren(bool
relayoutChildren, WebCore::LayoutUnit & maxFloatLogicalBottom) Line 632
C++
WebKit.dll!WebCore::RenderBlockFlow::layoutBlock(bool
relayoutChildren, WebCore::LayoutUnit pageLogicalHeight) Line 485 C++
WebKit.dll!WebCore::RenderBlock::layout() Line 930 C++
WebKit.dll!WebCore::RenderView::layoutContent(const
WebCore::LayoutState & state) Line 256 C++
WebKit.dll!WebCore::RenderView::layout() Line 382 C++
WebKit.dll!WebCore::FrameView::layout(bool allowSubtree) Line
1429 C++
WebKit.dll!WebCore::Document::updateLayout() Line 1889 C++
WebKit.dll!WebCore::Document::updateLayoutIgnorePendingStylesheets(WebCore::Document::RunPostLayoutTasks
runPostLayoutTasks) Line 1922 C++
WebKit.dll!WebCore::Element::getBoundingClientRect() Line 1071
C++
WebKit.dll!WebCore::jsElementPrototypeFunctionGetBoundingClientRect(JSC::ExecState
* exec) Line 5165 C++
[External Code]
JavaScriptCore.dll!llint_entry() Line 8217 Unknown
[External Code]
JavaScriptCore.dll!llint_entry() Line 8217 Unknown
From: Vienneau, Christopher
Sent: Friday, November 20, 2015 5:20 PM
To: 'WebKit Development' <[email protected]>
Subject: RE: Memory leak tracking in WebKit
To add some more information to this, the allocation that seems to have the
largest delta amount of allocations between test iterations is:
\wtf\local\wtf\hashtable.h(1248):
WTF::HashTable<WTF::RefPtr<WTF::UniquedStringImpl>,WTF::KeyValuePair<WTF::RefPtr<WTF::UniquedStringImpl>,JSC::VariableEnvironmentEntry>,WTF::KeyValuePairKeyExtractor<WTF::KeyValuePair<WTF::RefPtr<WTF::UniquedStringImpl>,JSC::VariableEnvironmentEntry>
>,JSC::IdentifierRepHash,WTF::HashMap<WTF::RefPtr<WTF::UniquedStringImpl>,JSC::VariableEnvironmentEntry,JSC::IdentifierRepHash,WTF::HashTraits<WTF::RefPtr<WTF::UniquedStringImpl>
>,JSC::VariableEnvironmentEntryHashTraits>::KeyValuePairTraits,WTF::HashTraits<WTF::RefPtr<WTF::UniquedStringImpl>
>
>::HashTable<WTF::RefPtr<WTF::UniquedStringImpl>,WTF::KeyValuePair<WTF::RefPtr<WTF::UniquedStringImpl>,JSC::VariableEnvironmentEntry>,WTF::KeyValuePairKeyExtractor<WTF::KeyValuePair<WTF::RefPtr<WTF::UniquedStringImpl>,JSC::VariableEnvironmentEntry>
>,JSC::IdentifierRepHash,WTF::HashMap<WTF::RefPtr<WTF::UniquedStringImpl>,JSC::VariableEnvironmentEntry,JSC::IdentifierRepHash,WTF::HashTraits<WTF::RefPtr<WTF::UniquedStringImpl>
>,JSC::VariableEnvironmentEntryHashTraits>::KeyValuePairTraits,WTF::HashTraits<WTF::RefPtr<WTF::UniquedStringImpl>
> > + 153
\javascriptcore\local\javascriptcore\parser\parser.cpp(1347):
JSC::Parser<JSC::Lexer<unsigned char> >::parseTryStatement<JSC::ASTBuilder> +
1984
\javascriptcore\local\javascriptcore\parser\parser.cpp(1480):
JSC::Parser<JSC::Lexer<unsigned char> >::parseStatement<JSC::ASTBuilder> + 671
\javascriptcore\local\javascriptcore\parser\parser.cpp(497):
JSC::Parser<JSC::Lexer<unsigned char>
>::parseStatementListItem<JSC::ASTBuilder> + 100
\javascriptcore\local\javascriptcore\parser\parser.cpp(379):
JSC::Parser<JSC::Lexer<unsigned char> >::parseSourceElements<JSC::ASTBuilder> +
745
\javascriptcore\local\javascriptcore\parser\parser.cpp(1406):
JSC::Parser<JSC::Lexer<unsigned char> >::parseBlockStatement<JSC::ASTBuilder> +
741
\javascriptcore\local\javascriptcore\parser\parser.cpp(1433):
JSC::Parser<JSC::Lexer<unsigned char> >::parseStatement<JSC::ASTBuilder> + 179
\javascriptcore\local\javascriptcore\parser\parser.cpp(497):
JSC::Parser<JSC::Lexer<unsigned char>
>::parseStatementListItem<JSC::ASTBuilder> + 100
\javascriptcore\local\javascriptcore\parser\parser.cpp(379):
JSC::Parser<JSC::Lexer<unsigned char> >::parseSourceElements<JSC::ASTBuilder> +
171
\javascriptcore\local\javascriptcore\parser\parser.cpp(277):
JSC::Parser<JSC::Lexer<unsigned char> >::parseInner + 654
To me this likely indicates this is what is leaking, does this bring up any
ideas for anyone?
Chris
From: Vienneau, Christopher
Sent: Thursday, November 19, 2015 10:25 PM
To: WebKit Development
<[email protected]<mailto:[email protected]>>
Subject: Memory leak tracking in WebKit
Hi,
I'm currently trying to track a leak in our port of WebKit. If left to soak in
an automated test, looping over 100 websites, visiting each for 10 seconds, for
an evening, I'm finding that memory usage goes up to well over a gigabyte
(eventually crashing). I know that a lot of things could remain cached but I'm
calling code very similar to
MemoryPressureHandler::releaseMemory(Critical::Yes, Synchronous::Yes) before
checking the memory counters built into our application. Manually trying to
reproduce the leak, I find that many pages don't seem to exhibit any
identifiable increase in memory when; starting from a simple page, clearing the
cache, visiting the test page, returning to the simple page and finally
clearing the cache. I have identified some pages which seem to be causing a
problem such as cnn.com, when visiting this page it appears I can expect to
lose 100-500k.
Our current code is based off of 188436:
-Are there any known memory leaks that I should pick up a fix for? (I searched
but nothing seemed relevant to the current code I have).
-Are there any tools or techniques that can be recommended for identifying a
leak? I've been improving our debug features of our memory system, but I've
been unable to solidly identify where the memory is going.
-I have the WinCairo sample built at the same revision, are there any memory
tools that can be used with it to identify if the leak exists there too?
Thanks for any suggestions
Chris Vienneau
/*
* Copyright (C) 2015 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "WebKitMessageLoop.h"
#include "WebKitDLL.h"
#if USE(GLIB)
#include <glib.h>
#endif
WebKitMessageLoop::WebKitMessageLoop()
: m_refCount(0)
{
gClassCount++;
gClassNameCount().add("WebKitMessageLoop");
}
WebKitMessageLoop::~WebKitMessageLoop()
{
gClassCount--;
gClassNameCount().remove("WebKitMessageLoop");
}
WebKitMessageLoop* WebKitMessageLoop::createInstance()
{
WebKitMessageLoop* instance = new WebKitMessageLoop();
instance->AddRef();
return instance;
}
HRESULT WebKitMessageLoop::QueryInterface(REFIID riid, void** ppvObject)
{
*ppvObject = 0;
if (IsEqualGUID(riid, IID_IUnknown))
*ppvObject = static_cast<IWebKitMessageLoop*>(this);
else if (IsEqualGUID(riid, CLSID_WebKitMessageLoop))
*ppvObject = static_cast<WebKitMessageLoop*>(this);
else if (IsEqualGUID(riid, IID_IWebKitMessageLoop))
*ppvObject = static_cast<IWebKitMessageLoop*>(this);
else
return E_NOINTERFACE;
AddRef();
return S_OK;
}
ULONG WebKitMessageLoop::AddRef()
{
return ++m_refCount;
}
ULONG WebKitMessageLoop::Release()
{
ULONG newRef = --m_refCount;
if (!newRef)
delete(this);
return newRef;
}
HRESULT WebKitMessageLoop::run(HACCEL hAccelTable)
{
MSG msg = { 0 };
while (GetMessage(&msg, 0, 0, 0)) {
performMessageLoopTasks();
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg)) {
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
return S_OK;
}
HRESULT WebKitMessageLoop::performMessageLoopTasks()
{
#if USE(CF)
CFRunLoopRunInMode(kCFRunLoopDefaultMode, 0, true);
#endif
#if USE(GLIB)
g_main_context_iteration(0, false);
#endif
//sneak my own message for our soak code to do what it wants
periodically
static DWORD lastTime = 0;
const UINT messageId = WM_USER + 0xEA;
if (GetCurrentTime() > (lastTime + 100)) // don't spam it too
much
{
lastTime = GetCurrentTime();
HWND windowHandle = FindWindow(NULL, TEXT("WinLauncher"));
PostMessage(windowHandle, messageId, 0, GetCurrentTime());
}
return S_OK;
}
/*
* Copyright (C) 2006, 2008, 2013-2015 Apple Inc. All rights reserved.
* Copyright (C) 2009, 2011 Brent Fulgham. All rights reserved.
* Copyright (C) 2009, 2010, 2011 Appcelerator, Inc. All rights reserved.
* Copyright (C) 2013 Alex Christensen. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "AccessibilityDelegate.h"
#include "DOMDefaultImpl.h"
#include "PrintWebUIDelegate.h"
#include "ResourceLoadDelegate.h"
#include "WebDownloadDelegate.h"
#include "WinLauncher.h"
#include "WinLauncherReplace.h"
#include <WebKit/WebKitCOMAPI.h>
#include <wtf/ExportMacros.h>
#include <wtf/Platform.h>
#include <wtf/text/CString.h>
#include <wtf/text/WTFString.h>
#if USE(CF)
#include <CoreFoundation/CFRunLoop.h>
#include <WebKit/CFDictionaryPropertyBag.h>
#endif
#include <cassert>
#include <comip.h>
#include <commctrl.h>
#include <commdlg.h>
#include <comutil.h>
#include <dbghelp.h>
#include <memory>
#include <objbase.h>
#include <shellapi.h>
#include <shlobj.h>
#include <shlwapi.h>
#include <string>
#include <vector>
#include <wininet.h>
#define MAX_LOADSTRING 100
#define URLBAR_HEIGHT 24
#define CONTROLBUTTON_WIDTH 24
static const int maxHistorySize = 10;
#ifndef WM_DPICHANGED
#define WM_DPICHANGED 0x02E0
#endif
typedef _com_ptr_t<_com_IIID<IWebFrame, &__uuidof(IWebFrame)>> IWebFramePtr;
typedef _com_ptr_t<_com_IIID<IWebMutableURLRequest,
&__uuidof(IWebMutableURLRequest)>> IWebMutableURLRequestPtr;
// Global Variables:
HINSTANCE hInst;
HWND hMainWnd;
HWND hURLBarWnd;
HGDIOBJ hURLFont;
HWND hBackButtonWnd;
HWND hForwardButtonWnd;
HWND hCacheWnd;
WNDPROC DefEditProc = nullptr;
WNDPROC DefButtonProc = nullptr;
WNDPROC DefWebKitProc = nullptr;
HWND gViewWindow = 0;
WinLauncher* gWinLauncher = nullptr;
TCHAR szTitle[MAX_LOADSTRING]; // The title bar text
TCHAR szWindowClass[MAX_LOADSTRING]; // the main window class name
// Support moving the transparent window
POINT s_windowPosition = { 100, 100 };
SIZE s_windowSize = { 800, 400 };
// Forward declarations of functions included in this code module:
ATOM MyRegisterClass(HINSTANCE hInstance);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK CustomUserAgent(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK EditProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK BackButtonProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK ForwardButtonProc(HWND, UINT, WPARAM, LPARAM);
LRESULT CALLBACK ReloadButtonProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK Caches(HWND, UINT, WPARAM, LPARAM);
static void loadURL(BSTR urlBStr);
static void updateStatistics(HWND hDlg);
namespace WebCore {
float deviceScaleFactorForWindow(HWND);
}
static void resizeSubViews()
{
if (gWinLauncher->usesLayeredWebView() || !gViewWindow)
return;
float scaleFactor = WebCore::deviceScaleFactorForWindow(gViewWindow);
RECT rcClient;
GetClientRect(hMainWnd, &rcClient);
int height = scaleFactor * URLBAR_HEIGHT;
int width = scaleFactor * CONTROLBUTTON_WIDTH;
MoveWindow(hBackButtonWnd, 0, 0, width, height, TRUE);
MoveWindow(hForwardButtonWnd, width, 0, width, height, TRUE);
MoveWindow(hURLBarWnd, width * 2, 0, rcClient.right, height, TRUE);
MoveWindow(gViewWindow, 0, height, rcClient.right, rcClient.bottom -
height, TRUE);
::SendMessage(hURLBarWnd, static_cast<UINT>(WM_SETFONT),
reinterpret_cast<WPARAM>(gWinLauncher->urlBarFont()), TRUE);
}
static void subclassForLayeredWindow()
{
hMainWnd = gViewWindow;
#if defined _M_AMD64 || defined _WIN64
DefWebKitProc = reinterpret_cast<WNDPROC>(::GetWindowLongPtr(hMainWnd,
GWLP_WNDPROC));
::SetWindowLongPtr(hMainWnd, GWLP_WNDPROC,
reinterpret_cast<LONG_PTR>(WndProc));
#else
DefWebKitProc = reinterpret_cast<WNDPROC>(::GetWindowLong(hMainWnd,
GWL_WNDPROC));
::SetWindowLong(hMainWnd, GWL_WNDPROC, reinterpret_cast<LONG_PTR>(WndProc));
#endif
}
static void computeFullDesktopFrame()
{
RECT desktop;
if (!::SystemParametersInfo(SPI_GETWORKAREA, 0,
static_cast<void*>(&desktop), 0))
return;
float scaleFactor = WebCore::deviceScaleFactorForWindow(nullptr);
s_windowPosition.x = 0;
s_windowPosition.y = 0;
s_windowSize.cx = scaleFactor * (desktop.right - desktop.left);
s_windowSize.cy = scaleFactor * (desktop.bottom - desktop.top);
}
BOOL WINAPI DllMain(HINSTANCE dllInstance, DWORD reason, LPVOID)
{
if (reason == DLL_PROCESS_ATTACH) {
#if defined(_M_X64) || defined(__x86_64__)
// The VS2013 runtime has a bug where it mis-detects AVX-capable
processors
// if the feature has been disabled in firmware. This causes us to crash
// in some of the math functions. For now, we disable those
optimizations
// because Microsoft is not going to fix the problem in VS2013.
// FIXME: http://webkit.org/b/141449: Remove this workaround when we
switch to VS2015+.
_set_FMA3_enable(0);
#endif
hInst = dllInstance;
}
return TRUE;
}
static bool getAppDataFolder(_bstr_t& directory)
{
wchar_t appDataDirectory[MAX_PATH];
if (FAILED(SHGetFolderPathW(0, CSIDL_LOCAL_APPDATA | CSIDL_FLAG_CREATE, 0,
0, appDataDirectory)))
return false;
wchar_t executablePath[MAX_PATH];
if (!::GetModuleFileNameW(0, executablePath, MAX_PATH))
return false;
::PathRemoveExtensionW(executablePath);
directory = _bstr_t(appDataDirectory) + L"\\" +
::PathFindFileNameW(executablePath);
return true;
}
static bool setCacheFolder()
{
IWebCachePtr webCache = gWinLauncher->webCache();
if (!webCache)
return false;
_bstr_t appDataFolder;
if (!getAppDataFolder(appDataFolder))
return false;
appDataFolder += L"\\cache";
webCache->setCacheFolder(appDataFolder);
return true;
}
void createCrashReport(EXCEPTION_POINTERS* exceptionPointers)
{
_bstr_t directory;
if (!getAppDataFolder(directory))
return;
if (::SHCreateDirectoryEx(0, directory, 0) != ERROR_SUCCESS
&& ::GetLastError() != ERROR_FILE_EXISTS
&& ::GetLastError() != ERROR_ALREADY_EXISTS)
return;
std::wstring fileName = directory + L"\\CrashReport.dmp";
HANDLE miniDumpFile = ::CreateFile(fileName.c_str(), GENERIC_WRITE, 0, 0,
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
if (miniDumpFile && miniDumpFile != INVALID_HANDLE_VALUE) {
MINIDUMP_EXCEPTION_INFORMATION mdei;
mdei.ThreadId = ::GetCurrentThreadId();
mdei.ExceptionPointers = exceptionPointers;
mdei.ClientPointers = 0;
#ifdef _DEBUG
MINIDUMP_TYPE dumpType = MiniDumpWithFullMemory;
#else
MINIDUMP_TYPE dumpType = MiniDumpNormal;
#endif
::MiniDumpWriteDump(::GetCurrentProcess(), ::GetCurrentProcessId(),
miniDumpFile, dumpType, &mdei, 0, 0);
::CloseHandle(miniDumpFile);
processCrashReport(fileName.c_str());
}
}
static BOOL CALLBACK AbortProc(HDC hDC, int Error)
{
MSG msg;
while (::PeekMessage(&msg, 0, 0, 0, PM_REMOVE)) {
::TranslateMessage(&msg);
::DispatchMessage(&msg);
}
return TRUE;
}
static HDC getPrinterDC()
{
PRINTDLG pdlg;
memset(&pdlg, 0, sizeof(PRINTDLG));
pdlg.lStructSize = sizeof(PRINTDLG);
pdlg.Flags = PD_PRINTSETUP | PD_RETURNDC;
::PrintDlg(&pdlg);
return pdlg.hDC;
}
static void initDocStruct(DOCINFO* di, TCHAR* docname)
{
memset(di, 0, sizeof(DOCINFO));
di->cbSize = sizeof(DOCINFO);
di->lpszDocName = docname;
}
typedef _com_ptr_t<_com_IIID<IWebFramePrivate, &__uuidof(IWebFramePrivate)>>
IWebFramePrivatePtr;
void PrintView(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
HDC printDC = getPrinterDC();
if (!printDC) {
::MessageBoxW(0, L"Error creating printing DC", L"Error", MB_APPLMODAL
| MB_OK);
return;
}
if (::SetAbortProc(printDC, AbortProc) == SP_ERROR) {
::MessageBoxW(0, L"Error setting up AbortProc", L"Error", MB_APPLMODAL
| MB_OK);
return;
}
IWebFramePtr frame = gWinLauncher->mainFrame();
if (!frame)
return;
IWebFramePrivatePtr framePrivate;
if (FAILED(frame->QueryInterface(&framePrivate.GetInterfacePtr())))
return;
framePrivate->setInPrintingMode(TRUE, printDC);
UINT pageCount = 0;
framePrivate->getPrintedPageCount(printDC, &pageCount);
DOCINFO di;
initDocStruct(&di, L"WebKit Doc");
::StartDoc(printDC, &di);
// FIXME: Need CoreGraphics implementation
void* graphicsContext = 0;
for (size_t page = 1; page <= pageCount; ++page) {
::StartPage(printDC);
framePrivate->spoolPages(printDC, page, page, graphicsContext);
::EndPage(printDC);
}
framePrivate->setInPrintingMode(FALSE, printDC);
::EndDoc(printDC);
::DeleteDC(printDC);
}
static void ToggleMenuFlag(HWND hWnd, UINT menuID)
{
HMENU menu = ::GetMenu(hWnd);
MENUITEMINFO info;
::memset(&info, 0x00, sizeof(info));
info.cbSize = sizeof(info);
info.fMask = MIIM_STATE;
if (!::GetMenuItemInfo(menu, menuID, FALSE, &info))
return;
BOOL newState = !(info.fState & MFS_CHECKED);
info.fState = (newState) ? MFS_CHECKED : MFS_UNCHECKED;
::SetMenuItemInfo(menu, menuID, FALSE, &info);
}
static bool menuItemIsChecked(const MENUITEMINFO& info)
{
return info.fState & MFS_CHECKED;
}
static void turnOffOtherUserAgents(HMENU menu)
{
MENUITEMINFO info;
::memset(&info, 0x00, sizeof(info));
info.cbSize = sizeof(info);
info.fMask = MIIM_STATE;
// Must unset the other menu items:
for (UINT menuToClear = IDM_UA_DEFAULT; menuToClear <= IDM_UA_OTHER;
++menuToClear) {
if (!::GetMenuItemInfo(menu, menuToClear, FALSE, &info))
continue;
if (!menuItemIsChecked(info))
continue;
info.fState = MFS_UNCHECKED;
::SetMenuItemInfo(menu, menuToClear, FALSE, &info);
}
}
static void ToggleMenuItem(HWND hWnd, UINT menuID)
{
HMENU menu = ::GetMenu(hWnd);
MENUITEMINFO info;
::memset(&info, 0x00, sizeof(info));
info.cbSize = sizeof(info);
info.fMask = MIIM_STATE;
if (!::GetMenuItemInfo(menu, menuID, FALSE, &info))
return;
BOOL newState = !menuItemIsChecked(info);
if (!gWinLauncher->standardPreferences() ||
!gWinLauncher->privatePreferences())
return;
switch (menuID) {
case IDM_AVFOUNDATION:
gWinLauncher->standardPreferences()->setAVFoundationEnabled(newState);
break;
case IDM_ACC_COMPOSITING:
gWinLauncher->privatePreferences()->setAcceleratedCompositingEnabled(newState);
break;
case IDM_WK_FULLSCREEN:
gWinLauncher->privatePreferences()->setFullScreenEnabled(newState);
break;
case IDM_COMPOSITING_BORDERS:
gWinLauncher->privatePreferences()->setShowDebugBorders(newState);
gWinLauncher->privatePreferences()->setShowRepaintCounter(newState);
break;
case IDM_DISABLE_IMAGES:
gWinLauncher->standardPreferences()->setLoadsImagesAutomatically(!newState);
break;
case IDM_DISABLE_STYLES:
gWinLauncher->privatePreferences()->setAuthorAndUserStylesEnabled(!newState);
break;
case IDM_DISABLE_JAVASCRIPT:
gWinLauncher->standardPreferences()->setJavaScriptEnabled(!newState);
break;
case IDM_DISABLE_LOCAL_FILE_RESTRICTIONS:
gWinLauncher->privatePreferences()->setAllowUniversalAccessFromFileURLs(newState);
gWinLauncher->privatePreferences()->setAllowFileAccessFromFileURLs(newState);
break;
case IDM_UA_DEFAULT:
case IDM_UA_SAFARI_8_0:
case IDM_UA_SAFARI_IOS_8_IPHONE:
case IDM_UA_SAFARI_IOS_8_IPAD:
case IDM_UA_IE_11:
case IDM_UA_CHROME_MAC:
case IDM_UA_CHROME_WIN:
case IDM_UA_FIREFOX_MAC:
case IDM_UA_FIREFOX_WIN:
gWinLauncher->setUserAgent(menuID);
turnOffOtherUserAgents(menu);
break;
case IDM_UA_OTHER:
// The actual user agent string will be set by the custom user agent
dialog
turnOffOtherUserAgents(menu);
break;
}
info.fState = (newState) ? MFS_CHECKED : MFS_UNCHECKED;
::SetMenuItemInfo(menu, menuID, FALSE, &info);
}
static const int dragBarHeight = 30;
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
WNDPROC parentProc = (gWinLauncher) ? (gWinLauncher->usesLayeredWebView() ?
DefWebKitProc : DefWindowProc) : DefWindowProc;
const UINT messageID = WM_USER + 0xEA;
switch (message) {
case messageID: {
static UINT lastTime = 0;
if (lParam > (lastTime + 10000)) //every 10 seconds
{
lastTime = lParam;
static UINT loadCount = 0;
loadCount++;
if ((loadCount %2) == 0)
{
loadURL(L"http://answers.ea.com");
}
else
{
loadURL(L"file:///C:/test_links.html");
}
}
break;
}
case WM_NCHITTEST:
if (gWinLauncher && gWinLauncher->usesLayeredWebView()) {
RECT window;
::GetWindowRect(hWnd, &window);
// For testing our transparent window, we need a region to use as a
handle for
// dragging. The right way to do this would be to query the web
view to see what's
// under the mouse. However, for testing purposes we just use an
arbitrary
// 30 logical pixel band at the top of the view as an arbitrary
gripping location.
//
// When we are within this bad, return HT_CAPTION to tell Windows
we want to
// treat this region as if it were the title bar on a normal window.
int y = HIWORD(lParam);
float scaledDragBarHeightFactor = dragBarHeight *
gWinLauncher->deviceScaleFactor();
if ((y > window.top) && (y < window.top +
scaledDragBarHeightFactor))
return HTCAPTION;
}
return CallWindowProc(parentProc, hWnd, message, wParam, lParam);
case WM_COMMAND: {
int wmId = LOWORD(wParam);
int wmEvent = HIWORD(wParam);
if (wmId >= IDM_HISTORY_LINK0 && wmId <= IDM_HISTORY_LINK9) {
if (gWinLauncher)
gWinLauncher->navigateToHistory(hWnd, wmId);
break;
}
// Parse the menu selections:
switch (wmId) {
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
DestroyWindow(hWnd);
break;
case IDM_PRINT:
PrintView(hWnd, message, wParam, lParam);
break;
case IDM_WEB_INSPECTOR:
if (gWinLauncher)
gWinLauncher->launchInspector();
break;
case IDM_CACHES:
if (!::IsWindow(hCacheWnd)) {
hCacheWnd = CreateDialog(hInst, MAKEINTRESOURCE(IDD_CACHES),
hWnd, Caches);
::ShowWindow(hCacheWnd, SW_SHOW);
}
break;
case IDM_HISTORY_BACKWARD:
case IDM_HISTORY_FORWARD:
if (gWinLauncher)
gWinLauncher->navigateForwardOrBackward(hWnd, wmId);
break;
case IDM_AVFOUNDATION:
case IDM_ACC_COMPOSITING:
case IDM_WK_FULLSCREEN:
case IDM_COMPOSITING_BORDERS:
case IDM_DISABLE_IMAGES:
case IDM_DISABLE_STYLES:
case IDM_DISABLE_JAVASCRIPT:
case IDM_DISABLE_LOCAL_FILE_RESTRICTIONS:
case IDM_UA_DEFAULT:
case IDM_UA_SAFARI_8_0:
case IDM_UA_SAFARI_IOS_8_IPHONE:
case IDM_UA_SAFARI_IOS_8_IPAD:
case IDM_UA_IE_11:
case IDM_UA_CHROME_MAC:
case IDM_UA_CHROME_WIN:
case IDM_UA_FIREFOX_MAC:
case IDM_UA_FIREFOX_WIN:
ToggleMenuItem(hWnd, wmId);
break;
case IDM_UA_OTHER:
if (wmEvent)
ToggleMenuItem(hWnd, wmId);
else
DialogBox(hInst, MAKEINTRESOURCE(IDD_USER_AGENT), hWnd,
CustomUserAgent);
break;
case IDM_ACTUAL_SIZE:
if (gWinLauncher)
gWinLauncher->resetZoom();
break;
case IDM_ZOOM_IN:
if (gWinLauncher)
gWinLauncher->zoomIn();
break;
case IDM_ZOOM_OUT:
if (gWinLauncher)
gWinLauncher->zoomOut();
break;
default:
return CallWindowProc(parentProc, hWnd, message, wParam, lParam);
}
}
break;
case WM_DESTROY:
#if USE(CF)
CFRunLoopStop(CFRunLoopGetMain());
#endif
PostQuitMessage(0);
break;
case WM_SIZE:
if (!gWinLauncher || !gWinLauncher->hasWebView() ||
gWinLauncher->usesLayeredWebView())
return CallWindowProc(parentProc, hWnd, message, wParam, lParam);
resizeSubViews();
break;
case WM_DPICHANGED:
if (gWinLauncher)
gWinLauncher->updateDeviceScaleFactor();
return CallWindowProc(parentProc, hWnd, message, wParam, lParam);
default:
return CallWindowProc(parentProc, hWnd, message, wParam, lParam);
}
return 0;
}
LRESULT CALLBACK EditProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message) {
case WM_CHAR:
if (wParam == 13) { // Enter Key
wchar_t strPtr[INTERNET_MAX_URL_LENGTH];
*((LPWORD)strPtr) = INTERNET_MAX_URL_LENGTH;
int strLen = SendMessage(hDlg, EM_GETLINE, 0, (LPARAM)strPtr);
strPtr[strLen] = 0;
_bstr_t bstr(strPtr);
loadURL(bstr.GetBSTR());
return 0;
}
default:
return CallWindowProc(DefEditProc, hDlg, message, wParam, lParam);
}
}
LRESULT CALLBACK BackButtonProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM
lParam)
{
switch (message) {
case WM_LBUTTONUP:
gWinLauncher->goBack();
default:
return CallWindowProc(DefButtonProc, hDlg, message, wParam, lParam);
}
}
LRESULT CALLBACK ForwardButtonProc(HWND hDlg, UINT message, WPARAM wParam,
LPARAM lParam)
{
switch (message) {
case WM_LBUTTONUP:
gWinLauncher->goForward();
default:
return CallWindowProc(DefButtonProc, hDlg, message, wParam, lParam);
}
}
// Message handler for about box.
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message) {
case WM_INITDIALOG:
return (INT_PTR)TRUE;
case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) {
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
INT_PTR CALLBACK Caches(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message) {
case WM_INITDIALOG:
::SetTimer(hDlg, IDT_UPDATE_STATS, 1000, nullptr);
return (INT_PTR)TRUE;
case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) {
::KillTimer(hDlg, IDT_UPDATE_STATS);
::DestroyWindow(hDlg);
hCacheWnd = 0;
return (INT_PTR)TRUE;
}
break;
case IDT_UPDATE_STATS:
::InvalidateRect(hDlg, nullptr, FALSE);
return (INT_PTR)TRUE;
case WM_PAINT:
updateStatistics(hDlg);
break;
}
return (INT_PTR)FALSE;
}
INT_PTR CALLBACK CustomUserAgent(HWND hDlg, UINT message, WPARAM wParam, LPARAM
lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message) {
case WM_INITDIALOG: {
HWND edit = ::GetDlgItem(hDlg, IDC_USER_AGENT_INPUT);
_bstr_t userAgent;
if (gWinLauncher)
userAgent = gWinLauncher->userAgent();
::SetWindowText(edit, static_cast<LPCTSTR>(userAgent));
return (INT_PTR)TRUE;
}
case WM_COMMAND:
if (LOWORD(wParam) == IDOK) {
HWND edit = ::GetDlgItem(hDlg, IDC_USER_AGENT_INPUT);
TCHAR buffer[1024];
int strLen = ::GetWindowText(edit, buffer, 1024);
buffer[strLen] = 0;
_bstr_t bstr(buffer);
if (bstr.length()) {
gWinLauncher->setUserAgent(bstr);
::PostMessage(hMainWnd, static_cast<UINT>(WM_COMMAND),
MAKELPARAM(IDM_UA_OTHER, 1), 0);
}
}
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL) {
::EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
static void loadURL(BSTR passedURL)
{
if (FAILED(gWinLauncher->loadURL(passedURL)))
return;
SetFocus(gViewWindow);
}
static void setWindowText(HWND dialog, UINT field, _bstr_t value)
{
::SetDlgItemText(dialog, field, value);
}
static void setWindowText(HWND dialog, UINT field, UINT value)
{
String valueStr = WTF::String::number(value);
setWindowText(dialog, field, _bstr_t(valueStr.utf8().data()));
}
typedef _com_ptr_t<_com_IIID<IPropertyBag, &__uuidof(IPropertyBag)>>
IPropertyBagPtr;
static void setWindowText(HWND dialog, UINT field, IPropertyBagPtr statistics,
const _bstr_t& key)
{
_variant_t var;
V_VT(&var) = VT_UI8;
if (FAILED(statistics->Read(key, &var.GetVARIANT(), nullptr)))
return;
unsigned long long value = V_UI8(&var);
String valueStr = WTF::String::number(value);
setWindowText(dialog, field, _bstr_t(valueStr.utf8().data()));
}
static void setWindowText(HWND dialog, UINT field, CFDictionaryRef dictionary,
CFStringRef key, UINT& total)
{
CFNumberRef countNum =
static_cast<CFNumberRef>(CFDictionaryGetValue(dictionary, key));
if (!countNum)
return;
int count = 0;
CFNumberGetValue(countNum, kCFNumberIntType, &count);
setWindowText(dialog, field, static_cast<UINT>(count));
total += count;
}
static void updateStatistics(HWND dialog)
{
if (!gWinLauncher)
return;
IWebCoreStatisticsPtr webCoreStatistics = gWinLauncher->statistics();
if (!webCoreStatistics)
return;
IPropertyBagPtr statistics;
HRESULT hr =
webCoreStatistics->memoryStatistics(&statistics.GetInterfacePtr());
if (FAILED(hr))
return;
// FastMalloc.
setWindowText(dialog, IDC_RESERVED_VM, statistics,
"FastMallocReservedVMBytes");
setWindowText(dialog, IDC_COMMITTED_VM, statistics,
"FastMallocCommittedVMBytes");
setWindowText(dialog, IDC_FREE_LIST_BYTES, statistics,
"FastMallocFreeListBytes");
// WebCore Cache.
#if USE(CF)
IWebCachePtr webCache = gWinLauncher->webCache();
int dictCount = 6;
IPropertyBag* cacheDict[6] = { 0 };
if (FAILED(webCache->statistics(&dictCount, cacheDict)))
return;
COMPtr<CFDictionaryPropertyBag> counts, sizes, liveSizes, decodedSizes,
purgableSizes;
counts.adoptRef(reinterpret_cast<CFDictionaryPropertyBag*>(cacheDict[0]));
sizes.adoptRef(reinterpret_cast<CFDictionaryPropertyBag*>(cacheDict[1]));
liveSizes.adoptRef(reinterpret_cast<CFDictionaryPropertyBag*>(cacheDict[2]));
decodedSizes.adoptRef(reinterpret_cast<CFDictionaryPropertyBag*>(cacheDict[3]));
purgableSizes.adoptRef(reinterpret_cast<CFDictionaryPropertyBag*>(cacheDict[4]));
static CFStringRef imagesKey = CFSTR("images");
static CFStringRef stylesheetsKey = CFSTR("style sheets");
static CFStringRef xslKey = CFSTR("xsl");
static CFStringRef scriptsKey = CFSTR("scripts");
if (counts) {
UINT totalObjects = 0;
setWindowText(dialog, IDC_IMAGES_OBJECT_COUNT, counts->dictionary(),
imagesKey, totalObjects);
setWindowText(dialog, IDC_CSS_OBJECT_COUNT, counts->dictionary(),
stylesheetsKey, totalObjects);
setWindowText(dialog, IDC_XSL_OBJECT_COUNT, counts->dictionary(),
xslKey, totalObjects);
setWindowText(dialog, IDC_JSC_OBJECT_COUNT, counts->dictionary(),
scriptsKey, totalObjects);
setWindowText(dialog, IDC_TOTAL_OBJECT_COUNT, totalObjects);
}
if (sizes) {
UINT totalBytes = 0;
setWindowText(dialog, IDC_IMAGES_BYTES, sizes->dictionary(), imagesKey,
totalBytes);
setWindowText(dialog, IDC_CSS_BYTES, sizes->dictionary(),
stylesheetsKey, totalBytes);
setWindowText(dialog, IDC_XSL_BYTES, sizes->dictionary(), xslKey,
totalBytes);
setWindowText(dialog, IDC_JSC_BYTES, sizes->dictionary(), scriptsKey,
totalBytes);
setWindowText(dialog, IDC_TOTAL_BYTES, totalBytes);
}
if (liveSizes) {
UINT totalLiveBytes = 0;
setWindowText(dialog, IDC_IMAGES_LIVE_COUNT, liveSizes->dictionary(),
imagesKey, totalLiveBytes);
setWindowText(dialog, IDC_CSS_LIVE_COUNT, liveSizes->dictionary(),
stylesheetsKey, totalLiveBytes);
setWindowText(dialog, IDC_XSL_LIVE_COUNT, liveSizes->dictionary(),
xslKey, totalLiveBytes);
setWindowText(dialog, IDC_JSC_LIVE_COUNT, liveSizes->dictionary(),
scriptsKey, totalLiveBytes);
setWindowText(dialog, IDC_TOTAL_LIVE_COUNT, totalLiveBytes);
}
if (decodedSizes) {
UINT totalDecoded = 0;
setWindowText(dialog, IDC_IMAGES_DECODED_COUNT,
decodedSizes->dictionary(), imagesKey, totalDecoded);
setWindowText(dialog, IDC_CSS_DECODED_COUNT,
decodedSizes->dictionary(), stylesheetsKey, totalDecoded);
setWindowText(dialog, IDC_XSL_DECODED_COUNT,
decodedSizes->dictionary(), xslKey, totalDecoded);
setWindowText(dialog, IDC_JSC_DECODED_COUNT,
decodedSizes->dictionary(), scriptsKey, totalDecoded);
setWindowText(dialog, IDC_TOTAL_DECODED, totalDecoded);
}
if (purgableSizes) {
UINT totalPurgable = 0;
setWindowText(dialog, IDC_IMAGES_PURGEABLE_COUNT,
purgableSizes->dictionary(), imagesKey, totalPurgable);
setWindowText(dialog, IDC_CSS_PURGEABLE_COUNT,
purgableSizes->dictionary(), stylesheetsKey, totalPurgable);
setWindowText(dialog, IDC_XSL_PURGEABLE_COUNT,
purgableSizes->dictionary(), xslKey, totalPurgable);
setWindowText(dialog, IDC_JSC_PURGEABLE_COUNT,
purgableSizes->dictionary(), scriptsKey, totalPurgable);
setWindowText(dialog, IDC_TOTAL_PURGEABLE, totalPurgable);
}
#endif
// JavaScript Heap.
setWindowText(dialog, IDC_JSC_HEAP_SIZE, statistics, "JavaScriptHeapSize");
setWindowText(dialog, IDC_JSC_HEAP_FREE, statistics, "JavaScriptFreeSize");
UINT count;
if (SUCCEEDED(webCoreStatistics->javaScriptObjectsCount(&count)))
setWindowText(dialog, IDC_TOTAL_JSC_HEAP_OBJECTS, count);
if (SUCCEEDED(webCoreStatistics->javaScriptGlobalObjectsCount(&count)))
setWindowText(dialog, IDC_GLOBAL_JSC_HEAP_OBJECTS, count);
if (SUCCEEDED(webCoreStatistics->javaScriptProtectedObjectsCount(&count)))
setWindowText(dialog, IDC_PROTECTED_JSC_HEAP_OBJECTS, count);
// Font and Glyph Caches.
if (SUCCEEDED(webCoreStatistics->cachedFontDataCount(&count)))
setWindowText(dialog, IDC_TOTAL_FONT_OBJECTS, count);
if (SUCCEEDED(webCoreStatistics->cachedFontDataInactiveCount(&count)))
setWindowText(dialog, IDC_INACTIVE_FONT_OBJECTS, count);
if (SUCCEEDED(webCoreStatistics->glyphPageCount(&count)))
setWindowText(dialog, IDC_GLYPH_PAGES, count);
// Site Icon Database.
if (SUCCEEDED(webCoreStatistics->iconPageURLMappingCount(&count)))
setWindowText(dialog, IDC_PAGE_URL_MAPPINGS, count);
if (SUCCEEDED(webCoreStatistics->iconRetainedPageURLCount(&count)))
setWindowText(dialog, IDC_RETAINED_PAGE_URLS, count);
if (SUCCEEDED(webCoreStatistics->iconRecordCount(&count)))
setWindowText(dialog, IDC_SITE_ICON_RECORDS, count);
if (SUCCEEDED(webCoreStatistics->iconsWithDataCount(&count)))
setWindowText(dialog, IDC_SITE_ICONS_WITH_DATA, count);
}
static void parseCommandLine(bool& usesLayeredWebView, bool& useFullDesktop,
bool& pageLoadTesting, _bstr_t& requestedURL)
{
usesLayeredWebView = false;
useFullDesktop = false;
pageLoadTesting = false;
int argc = 0;
WCHAR** argv = CommandLineToArgvW(GetCommandLineW(), &argc);
for (int i = 1; i < argc; ++i) {
if (!wcsicmp(argv[i], L"--transparent"))
usesLayeredWebView = true;
else if (!wcsicmp(argv[i], L"--desktop"))
useFullDesktop = true;
else if (!requestedURL)
requestedURL = argv[i];
else if (!wcsicmp(argv[i], L"--performance"))
pageLoadTesting = true;
}
}
extern "C" __declspec(dllexport) int WINAPI dllLauncherEntryPoint(HINSTANCE
hInstance, HINSTANCE hPrevInstance, LPWSTR lpstrCmdLine, int nCmdShow)
{
return wWinMain(hInstance, hPrevInstance, lpstrCmdLine, nCmdShow);
}
_______________________________________________
webkit-dev mailing list
[email protected]
https://lists.webkit.org/mailman/listinfo/webkit-dev