summaryrefslogtreecommitdiffstats
path: root/WebCore
diff options
context:
space:
mode:
authorSteve Block <steveblock@google.com>2009-09-02 18:13:59 +0100
committerSteve Block <steveblock@google.com>2009-09-04 18:03:58 +0100
commit4bb414d51f2850c0da64c0832fb0dbb0378a5707 (patch)
tree8b7c39f3150ae7e53db21cae12c075203e8ad2e9 /WebCore
parenta169a321cb5da710a16efc56e4186032b8b744c2 (diff)
downloadexternal_webkit-4bb414d51f2850c0da64c0832fb0dbb0378a5707.zip
external_webkit-4bb414d51f2850c0da64c0832fb0dbb0378a5707.tar.gz
external_webkit-4bb414d51f2850c0da64c0832fb0dbb0378a5707.tar.bz2
Fixes Geolocation to correctly handle infinite values for PositionOptions properties.
This fixes http://b/issue?id=2094429. Change-Id: I01cc5107d4a96840e35cc403161d50dbdf6a1ca2
Diffstat (limited to 'WebCore')
-rw-r--r--WebCore/bindings/js/JSGeolocationCustom.cpp22
-rw-r--r--WebCore/bindings/v8/custom/V8GeolocationCustom.cpp43
-rw-r--r--WebCore/page/PositionOptions.h12
3 files changed, 63 insertions, 14 deletions
diff --git a/WebCore/bindings/js/JSGeolocationCustom.cpp b/WebCore/bindings/js/JSGeolocationCustom.cpp
index 6379a1c..8ef8601 100644
--- a/WebCore/bindings/js/JSGeolocationCustom.cpp
+++ b/WebCore/bindings/js/JSGeolocationCustom.cpp
@@ -101,20 +101,34 @@ static PassRefPtr<PositionOptions> createPositionOptions(ExecState* exec, JSValu
if (exec->hadException())
return 0;
if (!timeoutValue.isUndefined()) {
- // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
- options->setTimeout(max(0, timeoutValue.toInt32(exec)));
+ double timeoutNumber = timeoutValue.toNumber(exec);
if (exec->hadException())
return 0;
+ // If the value is infinity, there's nothing to do.
+ if (timeoutNumber != Inf) {
+ // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
+ options->setTimeout(max(0, timeoutValue.toInt32(exec)));
+ if (exec->hadException())
+ return 0;
+ }
}
JSValue maximumAgeValue = object->get(exec, Identifier(exec, "maximumAge"));
if (exec->hadException())
return 0;
if (!maximumAgeValue.isUndefined()) {
- // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
- options->setMaximumAge(max(0, maximumAgeValue.toInt32(exec)));
+ double maximumAgeNumber = maximumAgeValue.toNumber(exec);
if (exec->hadException())
return 0;
+ if (maximumAgeNumber == Inf) {
+ // If the value is infinity, clear maximumAge.
+ options->clearMaximumAge();
+ } else {
+ // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
+ options->setMaximumAge(max(0, maximumAgeValue.toInt32(exec)));
+ if (exec->hadException())
+ return 0;
+ }
}
return options.release();
diff --git a/WebCore/bindings/v8/custom/V8GeolocationCustom.cpp b/WebCore/bindings/v8/custom/V8GeolocationCustom.cpp
index 5373626..7bc687c 100644
--- a/WebCore/bindings/v8/custom/V8GeolocationCustom.cpp
+++ b/WebCore/bindings/v8/custom/V8GeolocationCustom.cpp
@@ -38,6 +38,7 @@
#include "V8Proxy.h"
+using namespace std;
using namespace WTF;
namespace WebCore {
@@ -123,13 +124,28 @@ static PassRefPtr<PositionOptions> createPositionOptions(v8::Local<v8::Value> va
return 0;
}
if (!timeoutValue->IsUndefined()) {
- v8::Local<v8::Int32> timeoutInt32 = timeoutValue->ToInt32();
- if (timeoutInt32.IsEmpty()) {
+ v8::Local<v8::Number> timeoutNumber = timeoutValue->ToNumber();
+ if (timeoutNumber.IsEmpty()) {
succeeded = false;
return 0;
}
- // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
- options->setTimeout(max(0, timeoutInt32->Value()));
+ double timeoutDouble = timeoutNumber->Value();
+ // V8 does not export a public symbol for infinity, so we must use a
+ // platform type. On Android, it seems that V8 uses 0xf70f000000000000,
+ // which is the standard way to represent infinity in a double. However,
+ // numeric_limits<double>::infinity uses the system HUGE_VAL, which is
+ // different. Therefore we test using isinf() and check that the value
+ // is positive, which seems to handle things correctly.
+ // If the value is infinity, there's nothing to do.
+ if (!(isinf(timeoutDouble) && timeoutDouble > 0)) {
+ v8::Local<v8::Int32> timeoutInt32 = timeoutValue->ToInt32();
+ if (timeoutInt32.IsEmpty()) {
+ succeeded = false;
+ return 0;
+ }
+ // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
+ options->setTimeout(max(0, timeoutInt32->Value()));
+ }
}
v8::Local<v8::Value> maximumAgeValue = object->Get(v8::String::New("maximumAge"));
@@ -138,13 +154,24 @@ static PassRefPtr<PositionOptions> createPositionOptions(v8::Local<v8::Value> va
return 0;
}
if (!maximumAgeValue->IsUndefined()) {
- v8::Local<v8::Int32> maximumAgeInt32 = maximumAgeValue->ToInt32();
- if (maximumAgeInt32.IsEmpty()) {
+ v8::Local<v8::Number> maximumAgeNumber = maximumAgeValue->ToNumber();
+ if (maximumAgeNumber.IsEmpty()) {
succeeded = false;
return 0;
}
- // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
- options->setMaximumAge(max(0, maximumAgeInt32->Value()));
+ double maximumAgeDouble = maximumAgeNumber->Value();
+ if (isinf(maximumAgeDouble) && maximumAgeDouble > 0) {
+ // If the value is infinity, clear maximumAge.
+ options->clearMaximumAge();
+ } else {
+ v8::Local<v8::Int32> maximumAgeInt32 = maximumAgeValue->ToInt32();
+ if (maximumAgeInt32.IsEmpty()) {
+ succeeded = false;
+ return 0;
+ }
+ // Wrap to int32 and force non-negative to match behavior of window.setTimeout.
+ options->setMaximumAge(max(0, maximumAgeInt32->Value()));
+ }
}
return options.release();
diff --git a/WebCore/page/PositionOptions.h b/WebCore/page/PositionOptions.h
index ee8530a..5900998 100644
--- a/WebCore/page/PositionOptions.h
+++ b/WebCore/page/PositionOptions.h
@@ -49,10 +49,17 @@ public:
m_hasTimeout = true;
m_timeout = timeout;
}
- int maximumAge() const { return m_maximumAge; }
+ bool hasMaximumAge() const { return m_hasMaximumAge; }
+ int maximumAge() const
+ {
+ ASSERT(hasMaximumAge());
+ return m_maximumAge;
+ }
+ void clearMaximumAge() { m_hasMaximumAge = false; }
void setMaximumAge(int age)
{
ASSERT(age >= 0);
+ m_hasMaximumAge = true;
m_maximumAge = age;
}
@@ -60,13 +67,14 @@ private:
PositionOptions()
: m_highAccuracy(false)
, m_hasTimeout(false)
- , m_maximumAge(0)
{
+ setMaximumAge(0);
}
bool m_highAccuracy;
bool m_hasTimeout;
int m_timeout;
+ bool m_hasMaximumAge;
int m_maximumAge;
};