157 lines
5.1 KiB
C++
157 lines
5.1 KiB
C++
|
/*
|
||
|
* Copyright 2015 Google Inc.
|
||
|
*
|
||
|
* Use of this source code is governed by a BSD-style license that can be
|
||
|
* found in the LICENSE file.
|
||
|
*/
|
||
|
|
||
|
#include "SkTDPQueue.h"
|
||
|
#include "SkRandom.h"
|
||
|
#include "Test.h"
|
||
|
|
||
|
namespace { bool intless(const int& a, const int& b) { return a < b; } }
|
||
|
|
||
|
static void simple_test(skiatest::Reporter* reporter) {
|
||
|
SkTDPQueue<int, intless> heap;
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.count());
|
||
|
|
||
|
heap.insert(0);
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.count());
|
||
|
|
||
|
heap.insert(0);
|
||
|
heap.insert(1);
|
||
|
REPORTER_ASSERT(reporter, 2 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.count());
|
||
|
|
||
|
heap.insert(2);
|
||
|
heap.insert(1);
|
||
|
heap.insert(0);
|
||
|
REPORTER_ASSERT(reporter, 3 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 2 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 2 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.count());
|
||
|
|
||
|
heap.insert(2);
|
||
|
heap.insert(3);
|
||
|
heap.insert(0);
|
||
|
heap.insert(1);
|
||
|
REPORTER_ASSERT(reporter, 4 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 3 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 2 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 2 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 1 == heap.count());
|
||
|
REPORTER_ASSERT(reporter, 3 == heap.peek());
|
||
|
heap.pop();
|
||
|
REPORTER_ASSERT(reporter, 0 == heap.count());
|
||
|
}
|
||
|
|
||
|
struct Dummy {
|
||
|
int fValue;
|
||
|
int fPriority;
|
||
|
mutable int fIndex;
|
||
|
|
||
|
static bool LessP(Dummy* const& a, Dummy* const& b) { return a->fPriority < b->fPriority; }
|
||
|
static int* PQIndex(Dummy* const& dummy) { return &dummy->fIndex; }
|
||
|
|
||
|
bool operator== (const Dummy& that) const {
|
||
|
return fValue == that.fValue && fPriority == that.fPriority;
|
||
|
}
|
||
|
bool operator!= (const Dummy& that) const { return !(*this == that); }
|
||
|
};
|
||
|
|
||
|
void random_test(skiatest::Reporter* reporter) {
|
||
|
SkRandom random;
|
||
|
static const Dummy kSentinel = {-1, -1, -1};
|
||
|
|
||
|
for (int i = 0; i < 100; ++i) {
|
||
|
// Create a random set of Dummy objects.
|
||
|
int count = random.nextULessThan(100);
|
||
|
SkTDArray<Dummy> array;
|
||
|
array.setReserve(count);
|
||
|
for (int j = 0; j < count; ++j) {
|
||
|
Dummy* dummy = array.append();
|
||
|
dummy->fPriority = random.nextS();
|
||
|
dummy->fValue = random.nextS();
|
||
|
dummy->fIndex = -1;
|
||
|
if (*dummy == kSentinel) {
|
||
|
array.pop();
|
||
|
--j;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Stick the dummy objects in the pqueue.
|
||
|
SkTDPQueue<Dummy*, Dummy::LessP, Dummy::PQIndex> pq;
|
||
|
for (int j = 0; j < count; ++j) {
|
||
|
pq.insert(&array[j]);
|
||
|
}
|
||
|
REPORTER_ASSERT(reporter, pq.count() == array.count());
|
||
|
for (int j = 0; j < count; ++j) {
|
||
|
// every item should have an entry in the queue.
|
||
|
REPORTER_ASSERT(reporter, -1 != array[j].fIndex);
|
||
|
}
|
||
|
|
||
|
// Begin the test.
|
||
|
while (pq.count()) {
|
||
|
// Make sure the top of the queue is really the highest priority.
|
||
|
Dummy* top = pq.peek();
|
||
|
for (int k = 0; k < count; ++k) {
|
||
|
REPORTER_ASSERT(reporter, kSentinel == array[k] ||
|
||
|
array[k].fPriority >= top->fPriority);
|
||
|
}
|
||
|
// Do one of three random actions:
|
||
|
unsigned action = random.nextULessThan(3);
|
||
|
switch (action) {
|
||
|
case 0: { // pop the top,
|
||
|
Dummy* top = pq.peek();
|
||
|
REPORTER_ASSERT(reporter, array.begin() <= top && top < array.end());
|
||
|
pq.pop();
|
||
|
*top = kSentinel;
|
||
|
break;
|
||
|
}
|
||
|
case 1: { // remove a random element,
|
||
|
int item;
|
||
|
do {
|
||
|
item = random.nextULessThan(count);
|
||
|
} while (array[item] == kSentinel);
|
||
|
pq.remove(&array[item]);
|
||
|
array[item] = kSentinel;
|
||
|
break;
|
||
|
}
|
||
|
case 2: { // or change an element's priority.
|
||
|
int item;
|
||
|
do {
|
||
|
item = random.nextULessThan(count);
|
||
|
} while (array[item] == kSentinel);
|
||
|
array[item].fPriority = random.nextS();
|
||
|
pq.priorityDidChange(&array[item]);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DEF_TEST(TDPQueueTest, reporter) {
|
||
|
simple_test(reporter);
|
||
|
random_test(reporter);
|
||
|
}
|