Skip to content

Commit d05bc04

Browse files
committed
different approach
1 parent 23f8e28 commit d05bc04

File tree

9 files changed

+109
-99
lines changed

9 files changed

+109
-99
lines changed

include/boost/json/detail/impl/handler.ipp

+9-14
Original file line numberDiff line numberDiff line change
@@ -53,13 +53,8 @@ on_object_end(
5353
std::size_t n,
5454
error_code& ec)
5555
{
56-
if( !ignore_duplicate_keys )
57-
ec = st.check_duplicates(n);
58-
if( ec.failed() )
59-
return false;
60-
61-
st.push_object(n);
62-
return true;
56+
ec = st.push_object(n, ignore_duplicate_keys);
57+
return !ec.failed();
6358
}
6459

6560
bool
@@ -90,7 +85,7 @@ on_key_part(
9085
st.push_chars(s);
9186
return true;
9287
}
93-
88+
9489
bool
9590
handler::
9691
on_key(
@@ -101,12 +96,12 @@ on_key(
10196
st.push_key(s);
10297
return true;
10398
}
104-
99+
105100
bool
106101
handler::
107102
on_string_part(
108103
string_view s,
109-
std::size_t,
104+
std::size_t,
110105
error_code&)
111106
{
112107
st.push_chars(s);
@@ -117,7 +112,7 @@ bool
117112
handler::
118113
on_string(
119114
string_view s,
120-
std::size_t,
115+
std::size_t,
121116
error_code&)
122117
{
123118
st.push_string(s);
@@ -143,7 +138,7 @@ on_int64(
143138
st.push_int64(i);
144139
return true;
145140
}
146-
141+
147142
bool
148143
handler::
149144
on_uint64(
@@ -165,7 +160,7 @@ on_double(
165160
st.push_double(d);
166161
return true;
167162
}
168-
163+
169164
bool
170165
handler::
171166
on_bool(
@@ -192,7 +187,7 @@ on_comment_part(
192187
{
193188
return true;
194189
}
195-
190+
196191
bool
197192
handler::
198193
on_comment(

include/boost/json/detail/object.hpp

+20-15
Original file line numberDiff line numberDiff line change
@@ -28,30 +28,29 @@ class unchecked_object
2828
// first one is a string key,
2929
// second one is the value.
3030
value* data_;
31-
std::size_t size_;
31+
value* end_;
3232
storage_ptr const& sp_;
33+
bool ignore_duplicates_;
3334

3435
public:
3536
inline
3637
~unchecked_object();
3738

39+
inline
3840
unchecked_object(
3941
value* data,
4042
std::size_t size, // # of kv-pairs
41-
storage_ptr const& sp) noexcept
42-
: data_(data)
43-
, size_(size)
44-
, sp_(sp)
45-
{
46-
}
43+
storage_ptr const& sp,
44+
bool ignore_duplicates) noexcept;
4745

4846
unchecked_object(
4947
unchecked_object&& other) noexcept
5048
: data_(other.data_)
51-
, size_(other.size_)
49+
, end_(other.end_)
5250
, sp_(other.sp_)
51+
, ignore_duplicates_(other.ignore_duplicates_)
5352
{
54-
other.data_ = nullptr;
53+
other.data_ = other.end_ = nullptr;
5554
}
5655

5756
storage_ptr const&
@@ -60,19 +59,25 @@ class unchecked_object
6059
return sp_;
6160
}
6261

62+
inline
6363
std::size_t
64-
size() const noexcept
64+
size() const noexcept;
65+
66+
bool
67+
ignore_duplicate_keys() const noexcept
6568
{
66-
return size_;
69+
return ignore_duplicates_;
6770
}
6871

6972
value*
70-
release() noexcept
73+
front() noexcept
7174
{
72-
auto const data = data_;
73-
data_ = nullptr;
74-
return data;
75+
return data_;
7576
}
77+
78+
inline
79+
void
80+
pop_front() noexcept;
7681
};
7782

7883
template<class CharRange>

include/boost/json/impl/object.hpp

+29-5
Original file line numberDiff line numberDiff line change
@@ -518,19 +518,43 @@ namespace detail {
518518
unchecked_object::
519519
~unchecked_object()
520520
{
521-
if(! data_)
522-
return;
523521
if(sp_.is_not_shared_and_deallocate_is_trivial())
524522
return;
525-
value* p = data_;
526-
while(size_--)
523+
524+
for( value* p = data_; p != end_; p += 2 )
527525
{
528526
p[0].~value();
529527
p[1].~value();
530-
p += 2;
531528
}
532529
}
533530

531+
unchecked_object::
532+
unchecked_object(
533+
value* data,
534+
std::size_t size,
535+
storage_ptr const& sp,
536+
bool ignore_duplicates) noexcept
537+
: data_(data)
538+
, end_(data + 2 * size)
539+
, sp_(sp)
540+
, ignore_duplicates_(ignore_duplicates)
541+
{
542+
}
543+
544+
std::size_t
545+
unchecked_object::
546+
size() const noexcept
547+
{
548+
return std::size_t(end_ - data_) / 2;
549+
}
550+
551+
void
552+
unchecked_object::
553+
pop_front() noexcept
554+
{
555+
data_ += 2;
556+
}
557+
534558
} // detail
535559

536560
BOOST_JSON_NS_END

include/boost/json/impl/object.ipp

+18-11
Original file line numberDiff line numberDiff line change
@@ -197,7 +197,7 @@ destroy() noexcept
197197
//----------------------------------------------------------
198198

199199
object::
200-
object(detail::unchecked_object&& uo)
200+
object(detail::unchecked_object& uo)
201201
: sp_(uo.storage())
202202
{
203203
if(uo.size() == 0)
@@ -210,20 +210,18 @@ object(detail::unchecked_object&& uo)
210210
uo.size() <= max_size());
211211
t_ = table::allocate(
212212
uo.size(), 0, sp_);
213+
t_->size = 0;
213214

214215
// insert all elements, keeping
215-
// the last of any duplicate keys.
216+
// the last of any duplicate keys, unless uo.ignore_duplicates is false.
216217
auto dest = begin();
217-
auto src = uo.release();
218-
auto const end = src + 2 * uo.size();
219218
if(t_->is_small())
220219
{
221-
t_->size = 0;
222-
while(src != end)
220+
for( ; uo.size(); uo.pop_front() )
223221
{
222+
auto src = uo.front();
224223
access::construct_key_value_pair(
225224
dest, pilfer(src[0]), pilfer(src[1]));
226-
src += 2;
227225
auto result = detail::find_in_object(*this, dest->key());
228226
if(! result.first)
229227
{
@@ -232,6 +230,11 @@ object(detail::unchecked_object&& uo)
232230
continue;
233231
}
234232
// handle duplicate
233+
if( !uo.ignore_duplicate_keys() )
234+
{
235+
dest->~key_value_pair();
236+
return;
237+
}
235238
auto& v = *result.first;
236239
// don't bother to check if
237240
// storage deallocate is trivial
@@ -243,11 +246,11 @@ object(detail::unchecked_object&& uo)
243246
}
244247
return;
245248
}
246-
while(src != end)
249+
for( ; uo.size() ; uo.pop_front() )
247250
{
251+
auto src = uo.front();
248252
access::construct_key_value_pair(
249253
dest, pilfer(src[0]), pilfer(src[1]));
250-
src += 2;
251254
auto& head = t_->bucket(dest->key());
252255
auto i = head;
253256
for(;;)
@@ -260,6 +263,7 @@ object(detail::unchecked_object&& uo)
260263
head = static_cast<index_t>(
261264
dest - begin());
262265
++dest;
266+
++t_->size;
263267
break;
264268
}
265269
auto& v = (*t_)[i];
@@ -270,6 +274,11 @@ object(detail::unchecked_object&& uo)
270274
}
271275

272276
// handle duplicate
277+
if( !uo.ignore_duplicate_keys() )
278+
{
279+
dest->~key_value_pair();
280+
return;
281+
}
273282
access::next(*dest) =
274283
access::next(v);
275284
// don't bother to check if
@@ -282,8 +291,6 @@ object(detail::unchecked_object&& uo)
282291
break;
283292
}
284293
}
285-
t_->size = static_cast<
286-
index_t>(dest - begin());
287294
}
288295

289296
object::

include/boost/json/impl/value_stack.ipp

+14-43
Original file line numberDiff line numberDiff line change
@@ -82,37 +82,6 @@ has_chars()
8282
return chars_ != 0;
8383
}
8484

85-
error_code
86-
value_stack::
87-
stack::
88-
check_duplicates(std::size_t n)
89-
{
90-
error_code ec;
91-
92-
for( value* first = top_ - 2 * n; first != top_; first += 2 )
93-
{
94-
BOOST_ASSERT( first->is_string() );
95-
value* other = first + 2;
96-
while( true )
97-
{
98-
BOOST_ASSERT( other->is_string() );
99-
if( first->get_string() == other->get_string() )
100-
{
101-
BOOST_JSON_FAIL( ec, error::duplicate_key );
102-
goto before_return;
103-
}
104-
105-
if( other == top_ )
106-
break;
107-
108-
other += 2;
109-
}
110-
}
111-
112-
before_return:
113-
return ec;
114-
}
115-
11685
//--------------------------------------
11786

11887
// destroy the values but
@@ -320,7 +289,7 @@ exchange(Unchecked&& u)
320289
// which belongs to `u`.
321290
detail::access::
322291
construct_value(
323-
&jv.v, std::move(u));
292+
&jv.v, static_cast<Unchecked&&>(u));
324293
std::memcpy(
325294
reinterpret_cast<
326295
char*>(top_),
@@ -395,16 +364,25 @@ push_array(std::size_t n)
395364
st_.exchange(std::move(ua));
396365
}
397366

398-
void
367+
error_code
399368
value_stack::
400-
push_object(std::size_t n)
369+
push_object(std::size_t n, bool ignore_duplicates)
401370
{
402371
// we already have room if n > 0
403372
if(BOOST_JSON_UNLIKELY(n == 0))
404373
st_.maybe_grow();
374+
405375
detail::unchecked_object uo(
406-
st_.release(n * 2), n, sp_);
407-
st_.exchange(std::move(uo));
376+
st_.release(n * 2), n, sp_, ignore_duplicates);
377+
st_.exchange(uo);
378+
379+
error_code ec;
380+
// constructed object should have consumed all of uo's data
381+
if( uo.size() )
382+
{
383+
BOOST_JSON_FAIL( ec, error::duplicate_key );
384+
}
385+
return ec;
408386
}
409387

410388
void
@@ -498,13 +476,6 @@ push_null()
498476
st_.push(nullptr, sp_);
499477
}
500478

501-
error_code
502-
value_stack::
503-
check_duplicates(std::size_t n)
504-
{
505-
return st_.check_duplicates(n);
506-
}
507-
508479
BOOST_JSON_NS_END
509480

510481
#endif

include/boost/json/object.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,7 @@ class object
9090

9191
BOOST_JSON_DECL
9292
explicit
93-
object(detail::unchecked_object&& uo);
93+
object(detail::unchecked_object& uo);
9494

9595
public:
9696
/** The type of _Allocator_ returned by @ref get_allocator

include/boost/json/value.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -81,8 +81,8 @@ class value
8181

8282
explicit
8383
value(
84-
detail::unchecked_object&& uo)
85-
: obj_(std::move(uo))
84+
detail::unchecked_object& uo)
85+
: obj_(uo)
8686
{
8787
}
8888

0 commit comments

Comments
 (0)