put_by_val_direct need to check the property is index or not for using putDirect...
[WebKit-https.git] / Source / JavaScriptCore / runtime / PropertyName.h
1 /*
2  * Copyright (C) 2012 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. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #ifndef PropertyName_h
27 #define PropertyName_h
28
29 #include "Identifier.h"
30 #include "PrivateName.h"
31 #include <wtf/Optional.h>
32
33 namespace JSC {
34
35 template <typename CharType>
36 ALWAYS_INLINE Optional<uint32_t> toUInt32FromCharacters(const CharType* characters, unsigned length)
37 {
38     // An empty string is not a number.
39     if (!length)
40         return Nullopt;
41
42     // Get the first character, turning it into a digit.
43     uint32_t value = characters[0] - '0';
44     if (value > 9)
45         return Nullopt;
46     
47     // Check for leading zeros. If the first characher is 0, then the
48     // length of the string must be one - e.g. "042" is not equal to "42".
49     if (!value && length > 1)
50         return Nullopt;
51     
52     while (--length) {
53         // Multiply value by 10, checking for overflow out of 32 bits.
54         if (value > 0xFFFFFFFFU / 10)
55             return Nullopt;
56         value *= 10;
57         
58         // Get the next character, turning it into a digit.
59         uint32_t newValue = *(++characters) - '0';
60         if (newValue > 9)
61             return Nullopt;
62         
63         // Add in the old value, checking for overflow out of 32 bits.
64         newValue += value;
65         if (newValue < value)
66             return Nullopt;
67         value = newValue;
68     }
69
70     if (value == UINT_MAX)
71         return Nullopt;
72     return value;
73 }
74
75 ALWAYS_INLINE Optional<uint32_t> toUInt32FromStringImpl(StringImpl* impl)
76 {
77     if (impl->is8Bit())
78         return toUInt32FromCharacters(impl->characters8(), impl->length());
79     return toUInt32FromCharacters(impl->characters16(), impl->length());
80 }
81
82 class PropertyName {
83 public:
84     PropertyName(AtomicStringImpl* propertyName)
85         : m_impl(propertyName)
86     {
87         ASSERT(!m_impl || m_impl->isAtomic());
88     }
89
90     PropertyName(const Identifier& propertyName)
91         : PropertyName(static_cast<AtomicStringImpl*>(propertyName.impl()))
92     {
93     }
94
95     PropertyName(const PrivateName& propertyName)
96         : m_impl(static_cast<AtomicStringImpl*>(propertyName.uid()))
97     {
98         ASSERT(m_impl);
99         ASSERT(m_impl->isEmptyUnique());
100         ASSERT(m_impl->isAtomic());
101     }
102
103     AtomicStringImpl* uid() const
104     {
105         return m_impl;
106     }
107
108     AtomicStringImpl* publicName() const
109     {
110         return m_impl->isEmptyUnique() ? nullptr : m_impl;
111     }
112
113     static const uint32_t NotAnIndex = UINT_MAX;
114
115     Optional<uint32_t> asIndex()
116     {
117         return m_impl ? toUInt32FromStringImpl(m_impl) : Nullopt;
118     }
119
120     void dump(PrintStream& out) const
121     {
122         if (m_impl)
123             out.print(m_impl);
124         else
125             out.print("<null property name>");
126     }
127
128 private:
129     AtomicStringImpl* m_impl;
130 };
131
132 inline bool operator==(PropertyName a, const Identifier& b)
133 {
134     return a.uid() == b.impl();
135 }
136
137 inline bool operator==(const Identifier& a, PropertyName b)
138 {
139     return a.impl() == b.uid();
140 }
141
142 inline bool operator==(PropertyName a, PropertyName b)
143 {
144     return a.uid() == b.uid();
145 }
146
147 inline bool operator==(PropertyName a, const char* b)
148 {
149     return equal(a.uid(), b);
150 }
151
152 inline bool operator!=(PropertyName a, const Identifier& b)
153 {
154     return a.uid() != b.impl();
155 }
156
157 inline bool operator!=(const Identifier& a, PropertyName b)
158 {
159     return a.impl() != b.uid();
160 }
161
162 inline bool operator!=(PropertyName a, PropertyName b)
163 {
164     return a.uid() != b.uid();
165 }
166
167 }
168
169 #endif