• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <ctype.h>
6 #include <string>
7 
8 #include "base/compiler_specific.h"
9 #include "base/logging.h"
10 #include "base/memory/scoped_ptr.h"
11 #include "base/memory/scoped_vector.h"
12 #include "net/base/prioritized_dispatcher.h"
13 #include "net/base/request_priority.h"
14 #include "testing/gtest/include/gtest/gtest.h"
15 
16 namespace net {
17 
18 namespace {
19 
20 // We rely on the priority enum values being sequential having starting at 0,
21 // and increasing for higher priorities.
22 COMPILE_ASSERT(MINIMUM_PRIORITY == 0u &&
23                MINIMUM_PRIORITY == IDLE &&
24                IDLE < LOWEST &&
25                LOWEST < HIGHEST &&
26                HIGHEST <= MAXIMUM_PRIORITY,
27                priority_indexes_incompatible);
28 
29 class PrioritizedDispatcherTest : public testing::Test {
30  public:
31   typedef PrioritizedDispatcher::Priority Priority;
32   // A job that appends |tag| to |log| when started and '.' when finished.
33   // This is intended to confirm the execution order of a sequence of jobs added
34   // to the dispatcher. Note that finishing order of jobs does not matter.
35   class TestJob : public PrioritizedDispatcher::Job {
36    public:
TestJob(PrioritizedDispatcher * dispatcher,char tag,Priority priority,std::string * log)37     TestJob(PrioritizedDispatcher* dispatcher,
38             char tag,
39             Priority priority,
40             std::string* log)
41         : dispatcher_(dispatcher),
42           tag_(tag),
43           priority_(priority),
44           running_(false),
45           log_(log) {}
46 
running() const47     bool running() const {
48       return running_;
49     }
50 
handle() const51     const PrioritizedDispatcher::Handle handle() const {
52       return handle_;
53     }
54 
Add(bool at_head)55     void Add(bool at_head) {
56       CHECK(handle_.is_null());
57       CHECK(!running_);
58       size_t num_queued = dispatcher_->num_queued_jobs();
59       size_t num_running = dispatcher_->num_running_jobs();
60 
61       if (!at_head) {
62         handle_ = dispatcher_->Add(this, priority_);
63       } else {
64         handle_ = dispatcher_->AddAtHead(this, priority_);
65       }
66 
67       if (handle_.is_null()) {
68         EXPECT_EQ(num_queued, dispatcher_->num_queued_jobs());
69         EXPECT_TRUE(running_);
70         EXPECT_EQ(num_running + 1, dispatcher_->num_running_jobs());
71       } else {
72         EXPECT_FALSE(running_);
73         EXPECT_EQ(priority_, handle_.priority());
74         EXPECT_EQ(tag_, reinterpret_cast<TestJob*>(handle_.value())->tag_);
75         EXPECT_EQ(num_running, dispatcher_->num_running_jobs());
76       }
77     }
78 
ChangePriority(Priority priority)79     void ChangePriority(Priority priority) {
80       CHECK(!handle_.is_null());
81       CHECK(!running_);
82       size_t num_queued = dispatcher_->num_queued_jobs();
83       size_t num_running = dispatcher_->num_running_jobs();
84 
85       handle_ = dispatcher_->ChangePriority(handle_, priority);
86 
87       if (handle_.is_null()) {
88         EXPECT_TRUE(running_);
89         EXPECT_EQ(num_queued - 1, dispatcher_->num_queued_jobs());
90         EXPECT_EQ(num_running + 1, dispatcher_->num_running_jobs());
91       } else {
92         EXPECT_FALSE(running_);
93         EXPECT_EQ(priority, handle_.priority());
94         EXPECT_EQ(tag_, reinterpret_cast<TestJob*>(handle_.value())->tag_);
95         EXPECT_EQ(num_queued, dispatcher_->num_queued_jobs());
96         EXPECT_EQ(num_running, dispatcher_->num_running_jobs());
97       }
98     }
99 
Cancel()100     void Cancel() {
101       CHECK(!handle_.is_null());
102       CHECK(!running_);
103       size_t num_queued = dispatcher_->num_queued_jobs();
104 
105       dispatcher_->Cancel(handle_);
106 
107       EXPECT_EQ(num_queued - 1, dispatcher_->num_queued_jobs());
108       handle_ = PrioritizedDispatcher::Handle();
109     }
110 
Finish()111     void Finish() {
112       CHECK(running_);
113       running_ = false;
114       log_->append(1u, '.');
115 
116       dispatcher_->OnJobFinished();
117     }
118 
119     // PriorityDispatch::Job interface
Start()120     virtual void Start() OVERRIDE {
121       EXPECT_FALSE(running_);
122       handle_ = PrioritizedDispatcher::Handle();
123       running_ = true;
124       log_->append(1u, tag_);
125     }
126 
127    private:
128     PrioritizedDispatcher* dispatcher_;
129 
130     char tag_;
131     Priority priority_;
132 
133     PrioritizedDispatcher::Handle handle_;
134     bool running_;
135 
136     std::string* log_;
137   };
138 
139  protected:
Prepare(const PrioritizedDispatcher::Limits & limits)140   void Prepare(const PrioritizedDispatcher::Limits& limits) {
141     dispatcher_.reset(new PrioritizedDispatcher(limits));
142   }
143 
AddJob(char data,Priority priority)144   TestJob* AddJob(char data, Priority priority) {
145     TestJob* job = new TestJob(dispatcher_.get(), data, priority, &log_);
146     jobs_.push_back(job);
147     job->Add(false);
148     return job;
149   }
150 
AddJobAtHead(char data,Priority priority)151   TestJob* AddJobAtHead(char data, Priority priority) {
152     TestJob* job = new TestJob(dispatcher_.get(), data, priority, &log_);
153     jobs_.push_back(job);
154     job->Add(true);
155     return job;
156   }
157 
Expect(std::string log)158   void Expect(std::string log) {
159     EXPECT_EQ(0u, dispatcher_->num_queued_jobs());
160     EXPECT_EQ(0u, dispatcher_->num_running_jobs());
161     EXPECT_EQ(log, log_);
162     log_.clear();
163   }
164 
165   std::string log_;
166   scoped_ptr<PrioritizedDispatcher> dispatcher_;
167   ScopedVector<TestJob> jobs_;
168 };
169 
TEST_F(PrioritizedDispatcherTest,GetLimits)170 TEST_F(PrioritizedDispatcherTest, GetLimits) {
171   // Set non-trivial initial limits.
172   PrioritizedDispatcher::Limits original_limits(NUM_PRIORITIES, 5);
173   original_limits.reserved_slots[HIGHEST] = 1;
174   original_limits.reserved_slots[LOW] = 2;
175   Prepare(original_limits);
176 
177   // Get current limits, make sure the original limits are returned.
178   PrioritizedDispatcher::Limits retrieved_limits = dispatcher_->GetLimits();
179   ASSERT_EQ(original_limits.total_jobs, retrieved_limits.total_jobs);
180   ASSERT_EQ(NUM_PRIORITIES, retrieved_limits.reserved_slots.size());
181   for (size_t priority = MINIMUM_PRIORITY; priority <= MAXIMUM_PRIORITY;
182        ++priority) {
183     EXPECT_EQ(original_limits.reserved_slots[priority],
184               retrieved_limits.reserved_slots[priority]);
185   }
186 
187   // Set new limits.
188   PrioritizedDispatcher::Limits new_limits(NUM_PRIORITIES, 6);
189   new_limits.reserved_slots[MEDIUM] = 3;
190   new_limits.reserved_slots[LOWEST] = 1;
191   Prepare(new_limits);
192 
193   // Get current limits, make sure the new limits are returned.
194   retrieved_limits = dispatcher_->GetLimits();
195   ASSERT_EQ(new_limits.total_jobs, retrieved_limits.total_jobs);
196   ASSERT_EQ(NUM_PRIORITIES, retrieved_limits.reserved_slots.size());
197   for (size_t priority = MINIMUM_PRIORITY; priority <= MAXIMUM_PRIORITY;
198        ++priority) {
199     EXPECT_EQ(new_limits.reserved_slots[priority],
200               retrieved_limits.reserved_slots[priority]);
201   }
202 }
203 
TEST_F(PrioritizedDispatcherTest,AddAFIFO)204 TEST_F(PrioritizedDispatcherTest, AddAFIFO) {
205   // Allow only one running job.
206   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
207   Prepare(limits);
208 
209   TestJob* job_a = AddJob('a', IDLE);
210   TestJob* job_b = AddJob('b', IDLE);
211   TestJob* job_c = AddJob('c', IDLE);
212   TestJob* job_d = AddJob('d', IDLE);
213 
214   ASSERT_TRUE(job_a->running());
215   job_a->Finish();
216   ASSERT_TRUE(job_b->running());
217   job_b->Finish();
218   ASSERT_TRUE(job_c->running());
219   job_c->Finish();
220   ASSERT_TRUE(job_d->running());
221   job_d->Finish();
222 
223   Expect("a.b.c.d.");
224 }
225 
TEST_F(PrioritizedDispatcherTest,AddPriority)226 TEST_F(PrioritizedDispatcherTest, AddPriority) {
227   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
228   Prepare(limits);
229 
230   TestJob* job_a = AddJob('a', IDLE);
231   TestJob* job_b = AddJob('b', MEDIUM);
232   TestJob* job_c = AddJob('c', HIGHEST);
233   TestJob* job_d = AddJob('d', HIGHEST);
234   TestJob* job_e = AddJob('e', MEDIUM);
235 
236   ASSERT_TRUE(job_a->running());
237   job_a->Finish();
238   ASSERT_TRUE(job_c->running());
239   job_c->Finish();
240   ASSERT_TRUE(job_d->running());
241   job_d->Finish();
242   ASSERT_TRUE(job_b->running());
243   job_b->Finish();
244   ASSERT_TRUE(job_e->running());
245   job_e->Finish();
246 
247   Expect("a.c.d.b.e.");
248 }
249 
TEST_F(PrioritizedDispatcherTest,AddAtHead)250 TEST_F(PrioritizedDispatcherTest, AddAtHead) {
251   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
252   Prepare(limits);
253 
254   TestJob* job_a = AddJob('a', MEDIUM);
255   TestJob* job_b = AddJobAtHead('b', MEDIUM);
256   TestJob* job_c = AddJobAtHead('c', HIGHEST);
257   TestJob* job_d = AddJobAtHead('d', HIGHEST);
258   TestJob* job_e = AddJobAtHead('e', MEDIUM);
259   TestJob* job_f = AddJob('f', MEDIUM);
260 
261   ASSERT_TRUE(job_a->running());
262   job_a->Finish();
263   ASSERT_TRUE(job_d->running());
264   job_d->Finish();
265   ASSERT_TRUE(job_c->running());
266   job_c->Finish();
267   ASSERT_TRUE(job_e->running());
268   job_e->Finish();
269   ASSERT_TRUE(job_b->running());
270   job_b->Finish();
271   ASSERT_TRUE(job_f->running());
272   job_f->Finish();
273 
274   Expect("a.d.c.e.b.f.");
275 }
276 
TEST_F(PrioritizedDispatcherTest,EnforceLimits)277 TEST_F(PrioritizedDispatcherTest, EnforceLimits) {
278   // Reserve 2 for HIGHEST and 1 for LOW or higher.
279   // This leaves 2 for LOWEST or lower.
280   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 5);
281   limits.reserved_slots[HIGHEST] = 2;
282   limits.reserved_slots[LOW] = 1;
283   Prepare(limits);
284 
285   TestJob* job_a = AddJob('a', IDLE);    // Uses unreserved slot.
286   TestJob* job_b = AddJob('b', IDLE);    // Uses unreserved slot.
287   TestJob* job_c = AddJob('c', LOWEST);  // Must wait.
288   TestJob* job_d = AddJob('d', LOW);     // Uses reserved slot.
289   TestJob* job_e = AddJob('e', MEDIUM);  // Must wait.
290   TestJob* job_f = AddJob('f', HIGHEST); // Uses reserved slot.
291   TestJob* job_g = AddJob('g', HIGHEST); // Uses reserved slot.
292   TestJob* job_h = AddJob('h', HIGHEST); // Must wait.
293 
294   EXPECT_EQ(5u, dispatcher_->num_running_jobs());
295   EXPECT_EQ(3u, dispatcher_->num_queued_jobs());
296 
297   ASSERT_TRUE(job_a->running());
298   ASSERT_TRUE(job_b->running());
299   ASSERT_TRUE(job_d->running());
300   ASSERT_TRUE(job_f->running());
301   ASSERT_TRUE(job_g->running());
302   // a, b, d, f, g are running. Finish them in any order.
303   job_b->Finish();  // Releases h.
304   job_f->Finish();
305   job_a->Finish();
306   job_g->Finish();  // Releases e.
307   job_d->Finish();
308   ASSERT_TRUE(job_e->running());
309   ASSERT_TRUE(job_h->running());
310   // h, e are running.
311   job_e->Finish();  // Releases c.
312   ASSERT_TRUE(job_c->running());
313   job_c->Finish();
314   job_h->Finish();
315 
316   Expect("abdfg.h...e..c..");
317 }
318 
TEST_F(PrioritizedDispatcherTest,ChangePriority)319 TEST_F(PrioritizedDispatcherTest, ChangePriority) {
320   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 2);
321   // Reserve one slot only for HIGHEST priority requests.
322   limits.reserved_slots[HIGHEST] = 1;
323   Prepare(limits);
324 
325   TestJob* job_a = AddJob('a', IDLE);
326   TestJob* job_b = AddJob('b', LOW);
327   TestJob* job_c = AddJob('c', MEDIUM);
328   TestJob* job_d = AddJob('d', MEDIUM);
329   TestJob* job_e = AddJob('e', IDLE);
330 
331   ASSERT_FALSE(job_b->running());
332   ASSERT_FALSE(job_c->running());
333   job_b->ChangePriority(MEDIUM);
334   job_c->ChangePriority(LOW);
335 
336   ASSERT_TRUE(job_a->running());
337   job_a->Finish();
338   ASSERT_TRUE(job_d->running());
339   job_d->Finish();
340 
341   EXPECT_FALSE(job_e->running());
342   // Increasing |job_e|'s priority to HIGHEST should result in it being
343   // started immediately.
344   job_e->ChangePriority(HIGHEST);
345   ASSERT_TRUE(job_e->running());
346   job_e->Finish();
347 
348   ASSERT_TRUE(job_b->running());
349   job_b->Finish();
350   ASSERT_TRUE(job_c->running());
351   job_c->Finish();
352 
353   Expect("a.d.be..c.");
354 }
355 
TEST_F(PrioritizedDispatcherTest,Cancel)356 TEST_F(PrioritizedDispatcherTest, Cancel) {
357   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
358   Prepare(limits);
359 
360   TestJob* job_a = AddJob('a', IDLE);
361   TestJob* job_b = AddJob('b', IDLE);
362   TestJob* job_c = AddJob('c', IDLE);
363   TestJob* job_d = AddJob('d', IDLE);
364   TestJob* job_e = AddJob('e', IDLE);
365 
366   ASSERT_FALSE(job_b->running());
367   ASSERT_FALSE(job_d->running());
368   job_b->Cancel();
369   job_d->Cancel();
370 
371   ASSERT_TRUE(job_a->running());
372   job_a->Finish();
373   ASSERT_TRUE(job_c->running());
374   job_c->Finish();
375   ASSERT_TRUE(job_e->running());
376   job_e->Finish();
377 
378   Expect("a.c.e.");
379 }
380 
TEST_F(PrioritizedDispatcherTest,Evict)381 TEST_F(PrioritizedDispatcherTest, Evict) {
382   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
383   Prepare(limits);
384 
385   TestJob* job_a = AddJob('a', IDLE);
386   TestJob* job_b = AddJob('b', LOW);
387   TestJob* job_c = AddJob('c', HIGHEST);
388   TestJob* job_d = AddJob('d', LOW);
389   TestJob* job_e = AddJob('e', HIGHEST);
390 
391   EXPECT_EQ(job_b, dispatcher_->EvictOldestLowest());
392   EXPECT_EQ(job_d, dispatcher_->EvictOldestLowest());
393 
394   ASSERT_TRUE(job_a->running());
395   job_a->Finish();
396   ASSERT_TRUE(job_c->running());
397   job_c->Finish();
398   ASSERT_TRUE(job_e->running());
399   job_e->Finish();
400 
401   Expect("a.c.e.");
402 }
403 
TEST_F(PrioritizedDispatcherTest,EvictFromEmpty)404 TEST_F(PrioritizedDispatcherTest, EvictFromEmpty) {
405   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
406   Prepare(limits);
407   EXPECT_TRUE(dispatcher_->EvictOldestLowest() == NULL);
408 }
409 
TEST_F(PrioritizedDispatcherTest,AddWhileZeroLimits)410 TEST_F(PrioritizedDispatcherTest, AddWhileZeroLimits) {
411   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 2);
412   Prepare(limits);
413 
414   dispatcher_->SetLimitsToZero();
415   TestJob* job_a = AddJob('a', LOW);
416   TestJob* job_b = AddJob('b', MEDIUM);
417   TestJob* job_c = AddJobAtHead('c', MEDIUM);
418 
419   EXPECT_EQ(0u, dispatcher_->num_running_jobs());
420   EXPECT_EQ(3u, dispatcher_->num_queued_jobs());
421 
422   dispatcher_->SetLimits(limits);
423   EXPECT_EQ(2u, dispatcher_->num_running_jobs());
424   EXPECT_EQ(1u, dispatcher_->num_queued_jobs());
425 
426   ASSERT_TRUE(job_b->running());
427   job_b->Finish();
428 
429   ASSERT_TRUE(job_c->running());
430   job_c->Finish();
431 
432   ASSERT_TRUE(job_a->running());
433   job_a->Finish();
434 
435   Expect("cb.a..");
436 }
437 
TEST_F(PrioritizedDispatcherTest,ReduceLimitsWhileJobQueued)438 TEST_F(PrioritizedDispatcherTest, ReduceLimitsWhileJobQueued) {
439   PrioritizedDispatcher::Limits initial_limits(NUM_PRIORITIES, 2);
440   Prepare(initial_limits);
441 
442   TestJob* job_a = AddJob('a', MEDIUM);
443   TestJob* job_b = AddJob('b', MEDIUM);
444   TestJob* job_c = AddJob('c', MEDIUM);
445   TestJob* job_d = AddJob('d', MEDIUM);
446   TestJob* job_e = AddJob('e', MEDIUM);
447 
448   EXPECT_EQ(2u, dispatcher_->num_running_jobs());
449   EXPECT_EQ(3u, dispatcher_->num_queued_jobs());
450 
451   // Reduce limits to just allow one job at a time.  Running jobs should not
452   // be affected.
453   dispatcher_->SetLimits(PrioritizedDispatcher::Limits(NUM_PRIORITIES, 1));
454 
455   EXPECT_EQ(2u, dispatcher_->num_running_jobs());
456   EXPECT_EQ(3u, dispatcher_->num_queued_jobs());
457 
458   // Finishing a job should not result in another job starting.
459   ASSERT_TRUE(job_a->running());
460   job_a->Finish();
461   EXPECT_EQ(1u, dispatcher_->num_running_jobs());
462   EXPECT_EQ(3u, dispatcher_->num_queued_jobs());
463 
464   ASSERT_TRUE(job_b->running());
465   job_b->Finish();
466   EXPECT_EQ(1u, dispatcher_->num_running_jobs());
467   EXPECT_EQ(2u, dispatcher_->num_queued_jobs());
468 
469   // Increasing the limits again should let c start.
470   dispatcher_->SetLimits(initial_limits);
471 
472   ASSERT_TRUE(job_c->running());
473   job_c->Finish();
474   ASSERT_TRUE(job_d->running());
475   job_d->Finish();
476   ASSERT_TRUE(job_e->running());
477   job_e->Finish();
478 
479   Expect("ab..cd.e..");
480 }
481 
TEST_F(PrioritizedDispatcherTest,ZeroLimitsThenCancel)482 TEST_F(PrioritizedDispatcherTest, ZeroLimitsThenCancel) {
483   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
484   Prepare(limits);
485 
486   TestJob* job_a = AddJob('a', IDLE);
487   TestJob* job_b = AddJob('b', IDLE);
488   TestJob* job_c = AddJob('c', IDLE);
489   dispatcher_->SetLimitsToZero();
490 
491   ASSERT_TRUE(job_a->running());
492   EXPECT_FALSE(job_b->running());
493   EXPECT_FALSE(job_c->running());
494   job_a->Finish();
495 
496   EXPECT_FALSE(job_b->running());
497   EXPECT_FALSE(job_c->running());
498 
499   // Cancelling b shouldn't start job c.
500   job_b->Cancel();
501   EXPECT_FALSE(job_c->running());
502 
503   // Restoring the limits should start c.
504   dispatcher_->SetLimits(limits);
505   ASSERT_TRUE(job_c->running());
506   job_c->Finish();
507 
508   Expect("a.c.");
509 }
510 
TEST_F(PrioritizedDispatcherTest,ZeroLimitsThenIncreasePriority)511 TEST_F(PrioritizedDispatcherTest, ZeroLimitsThenIncreasePriority) {
512   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 2);
513   limits.reserved_slots[HIGHEST] = 1;
514   Prepare(limits);
515 
516   TestJob* job_a = AddJob('a', IDLE);
517   TestJob* job_b = AddJob('b', IDLE);
518   EXPECT_TRUE(job_a->running());
519   EXPECT_FALSE(job_b->running());
520   dispatcher_->SetLimitsToZero();
521 
522   job_b->ChangePriority(HIGHEST);
523   EXPECT_FALSE(job_b->running());
524   job_a->Finish();
525   EXPECT_FALSE(job_b->running());
526 
527   job_b->Cancel();
528   Expect("a.");
529 }
530 
531 #if GTEST_HAS_DEATH_TEST && !defined(NDEBUG)
TEST_F(PrioritizedDispatcherTest,CancelNull)532 TEST_F(PrioritizedDispatcherTest, CancelNull) {
533   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
534   Prepare(limits);
535   EXPECT_DEBUG_DEATH(dispatcher_->Cancel(PrioritizedDispatcher::Handle()), "");
536 }
537 
TEST_F(PrioritizedDispatcherTest,CancelMissing)538 TEST_F(PrioritizedDispatcherTest, CancelMissing) {
539   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
540   Prepare(limits);
541   AddJob('a', IDLE);
542   TestJob* job_b = AddJob('b', IDLE);
543   PrioritizedDispatcher::Handle handle = job_b->handle();
544   ASSERT_FALSE(handle.is_null());
545   dispatcher_->Cancel(handle);
546   EXPECT_DEBUG_DEATH(dispatcher_->Cancel(handle), "");
547 }
548 
549 // TODO(szym): Fix the PriorityQueue::Pointer check to die here.
550 // http://crbug.com/130846
TEST_F(PrioritizedDispatcherTest,DISABLED_CancelIncompatible)551 TEST_F(PrioritizedDispatcherTest, DISABLED_CancelIncompatible) {
552   PrioritizedDispatcher::Limits limits(NUM_PRIORITIES, 1);
553   Prepare(limits);
554   AddJob('a', IDLE);
555   TestJob* job_b = AddJob('b', IDLE);
556   PrioritizedDispatcher::Handle handle = job_b->handle();
557   ASSERT_FALSE(handle.is_null());
558 
559   // New dispatcher.
560   Prepare(limits);
561   AddJob('a', IDLE);
562   AddJob('b', IDLE);
563   EXPECT_DEBUG_DEATH(dispatcher_->Cancel(handle), "");
564 }
565 #endif  // GTEST_HAS_DEATH_TEST && !defined(NDEBUG)
566 
567 }  // namespace
568 
569 }  // namespace net
570