Only add prototypes to the PrototypeMap if they're not already present
[WebKit-https.git] / Source / JavaScriptCore / runtime / PrototypeMapInlines.h
1 /*
2  * Copyright (C) 2016 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #pragma once
27
28 #include "PrototypeMap.h"
29 #include "WeakGCMapInlines.h"
30
31 namespace JSC {
32
33 ALWAYS_INLINE TriState PrototypeMap::isPrototype(JSObject* object) const
34 {
35     if (!m_prototypes.contains(object))
36         return FalseTriState;
37
38     // We know that 'object' was used as a prototype at one time, so be
39     // conservative and say that it might still be so. (It would be expensive
40     // to find out for sure, and we don't know of any cases where being precise
41     // would improve performance.)
42     return MixedTriState;
43 }
44
45 ALWAYS_INLINE void PrototypeMap::addPrototype(JSObject* object)
46 {
47     auto addResult = m_prototypes.add(object, Weak<JSObject>());
48     if (addResult.isNewEntry)
49         addResult.iterator->value = Weak<JSObject>(object);
50     else
51         ASSERT(addResult.iterator->value.get() == object);
52
53     // Note that this method makes the somewhat odd decision to not check if this
54     // object currently has indexed accessors. We could do that check here, and if
55     // indexed accessors were found, we could tell the global object to have a bad
56     // time. But we avoid this, to allow the following to be always fast:
57     //
58     // 1) Create an object.
59     // 2) Give it a setter or read-only property that happens to have a numeric name.
60     // 3) Allocate objects that use this object as a prototype.
61     //
62     // This avoids anyone having a bad time. Even if the instance objects end up
63     // having indexed storage, the creation of indexed storage leads to a prototype
64     // chain walk that detects the presence of indexed setters and then does the
65     // right thing. As a result, having a bad time only happens if you add an
66     // indexed setter (or getter, or read-only field) to an object that is already
67     // used as a prototype.
68 }
69
70 } // namespace JSC
71