/*
* Copyright (C) 2006, 2007, 2008 Apple Inc. All Rights Reserved.
* Copyright (C) 2008 Collabora, Ltd. 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 "PluginMessageThrottlerWin.h"
#include "PluginView.h"
#include <wtf/ASCIICType.h>
#include <wtf/CurrentTime.h>
using namespace WTF;
namespace WebCore {
// Set a timer to make sure we process any queued messages at least every 16ms.
// This value allows Flash 60 messages/second, which should be enough for video
// playback, and also gets us over the limit for kicking into high-resolution
// timer mode (see SharedTimerWin.cpp).
static const double MessageThrottleTimeInterval = 0.016;
// During a continuous stream of messages, process one every 5ms.
static const double MessageDirectProcessingInterval = 0.005;
PluginMessageThrottlerWin::PluginMessageThrottlerWin(PluginView* pluginView)
: m_pluginView(pluginView)
, m_back(0)
, m_front(0)
, m_messageThrottleTimer(this, &PluginMessageThrottlerWin::messageThrottleTimerFired)
, m_lastMessageTime(0)
{
// Initialize the free list with our inline messages
for (unsigned i = 0; i < NumInlineMessages - 1; i++)
m_inlineMessages[i].next = &m_inlineMessages[i + 1];
m_inlineMessages[NumInlineMessages - 1].next = 0;
m_freeInlineMessages = &m_inlineMessages[0];
}
PluginMessageThrottlerWin::~PluginMessageThrottlerWin()
{
PluginMessage* next;
for (PluginMessage* message = m_front; message; message = next) {
next = message->next;
freeMessage(message);
}
}
void PluginMessageThrottlerWin::appendMessage(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
PluginMessage* message = allocateMessage();
message->hWnd = hWnd;
message->msg = msg;
message->wParam = wParam;
message->lParam = lParam;
message->next = 0;
if (m_back)
m_back->next = message;
m_back = message;
if (!m_front)
m_front = message;
// If it has been more than MessageDirectProcessingInterval between throttled messages,
// go ahead and process a message directly.
double currentTime = WTF::currentTime();
if (currentTime - m_lastMessageTime > MessageDirectProcessingInterval) {
processQueuedMessage();
m_lastMessageTime = currentTime;
if (!m_front)
return;
}
if (!m_messageThrottleTimer.isActive())
m_messageThrottleTimer.startOneShot(MessageThrottleTimeInterval);
}
void PluginMessageThrottlerWin::processQueuedMessage()
{
PluginMessage* message = m_front;
m_front = m_front->next;
if (message == m_back)
m_back = 0;
// Protect the PluginView from destruction while calling its window proc.
// <rdar://problem/6930280>
RefPtr<PluginView> protect(m_pluginView);
::CallWindowProc(m_pluginView->pluginWndProc(), message->hWnd, message->msg, message->wParam, message->lParam);
freeMessage(message);
}
void PluginMessageThrottlerWin::messageThrottleTimerFired(Timer<PluginMessageThrottlerWin>*)
{
processQueuedMessage();
if (m_front)
m_messageThrottleTimer.startOneShot(MessageThrottleTimeInterval);
}
PluginMessage* PluginMessageThrottlerWin::allocateMessage()
{
PluginMessage *message;
if (m_freeInlineMessages) {
message = m_freeInlineMessages;
m_freeInlineMessages = message->next;
} else
message = new PluginMessage;
return message;
}
bool PluginMessageThrottlerWin::isInlineMessage(PluginMessage* message)
{
return message >= &m_inlineMessages[0] && message <= &m_inlineMessages[NumInlineMessages - 1];
}
void PluginMessageThrottlerWin::freeMessage(PluginMessage* message)
{
if (isInlineMessage(message)) {
message->next = m_freeInlineMessages;
m_freeInlineMessages = message;
} else
delete message;
}
} // namespace WebCore