2019-09-19 22:42:45 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
|
|
|
Copyright (C) 2010-2019 EDuke32 developers and contributors
|
|
|
|
Copyright (C) 2019 Nuke.YKT
|
|
|
|
|
|
|
|
This file is part of NBlood.
|
|
|
|
|
|
|
|
NBlood is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License version 2
|
|
|
|
as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
#pragma once
|
|
|
|
#include <set>
|
|
|
|
#include <functional>
|
2019-07-13 17:50:23 +00:00
|
|
|
#include "common_game.h"
|
2019-09-22 06:39:22 +00:00
|
|
|
|
|
|
|
BEGIN_BLD_NS
|
|
|
|
|
2020-10-11 09:52:30 +00:00
|
|
|
enum { kPQueueSize = 1024 };
|
2019-09-19 22:42:45 +00:00
|
|
|
|
2019-07-13 17:50:23 +00:00
|
|
|
template <typename T> struct queueItem
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
uint32_t TotalKills; // priority
|
|
|
|
T Kills; // data
|
2019-09-19 22:42:45 +00:00
|
|
|
bool operator>(const queueItem& other) const
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return TotalKills > other.TotalKills;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
bool operator<(const queueItem& other) const
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return TotalKills < other.TotalKills;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
bool operator>=(const queueItem& other) const
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return TotalKills >= other.TotalKills;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
bool operator<=(const queueItem& other) const
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return TotalKills <= other.TotalKills;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
bool operator!=(const queueItem& other) const
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return TotalKills != other.TotalKills;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
bool operator==(const queueItem& other) const
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return TotalKills == other.TotalKills;
|
2019-09-19 22:42:45 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-07-13 17:50:23 +00:00
|
|
|
template<typename T> class PriorityQueue
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
|
|
|
public:
|
2019-07-13 17:50:23 +00:00
|
|
|
virtual ~PriorityQueue() {}
|
|
|
|
virtual uint32_t Size(void) = 0;
|
2019-09-19 22:42:45 +00:00
|
|
|
virtual void Clear(void) = 0;
|
2019-07-13 17:50:23 +00:00
|
|
|
virtual void Insert(uint32_t, T) = 0;
|
|
|
|
virtual T Remove(void) = 0;
|
|
|
|
virtual uint32_t LowestPriority(void) = 0;
|
|
|
|
virtual void Kill(std::function<bool(T)> pMatch) = 0;
|
2019-09-19 22:42:45 +00:00
|
|
|
};
|
|
|
|
|
2019-07-13 17:50:23 +00:00
|
|
|
template<typename T> class VanillaPriorityQueue : public PriorityQueue<T>
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
|
|
|
public:
|
2019-07-13 17:50:23 +00:00
|
|
|
queueItem<T> queueItems[kPQueueSize + 1];
|
|
|
|
uint32_t fNodeCount; // at2008
|
|
|
|
~VanillaPriorityQueue() {}
|
|
|
|
uint32_t Size(void) { return fNodeCount; };
|
|
|
|
void Clear(void)
|
|
|
|
{
|
|
|
|
fNodeCount = 0;
|
|
|
|
memset(queueItems, 0, sizeof(queueItems));
|
|
|
|
}
|
|
|
|
void Upheap(void)
|
|
|
|
{
|
|
|
|
queueItem<T> item = queueItems[fNodeCount];
|
2020-09-01 19:27:32 +00:00
|
|
|
queueItems[0].TotalKills = 0;
|
2019-07-13 17:50:23 +00:00
|
|
|
uint32_t x = fNodeCount;
|
|
|
|
while (queueItems[x>>1] > item)
|
|
|
|
{
|
|
|
|
queueItems[x] = queueItems[x>>1];
|
|
|
|
x >>= 1;
|
|
|
|
}
|
|
|
|
queueItems[x] = item;
|
|
|
|
}
|
|
|
|
void Downheap(uint32_t n)
|
|
|
|
{
|
|
|
|
queueItem<T> item = queueItems[n];
|
|
|
|
while (fNodeCount/2 >= n)
|
|
|
|
{
|
|
|
|
uint32_t t = n*2;
|
|
|
|
if (t < fNodeCount && queueItems[t] > queueItems[t+1])
|
|
|
|
t++;
|
|
|
|
if (item <= queueItems[t])
|
|
|
|
break;
|
|
|
|
queueItems[n] = queueItems[t];
|
|
|
|
n = t;
|
|
|
|
}
|
|
|
|
queueItems[n] = item;
|
|
|
|
}
|
|
|
|
void Delete(uint32_t k)
|
|
|
|
{
|
2020-10-11 10:38:17 +00:00
|
|
|
assert(k <= fNodeCount);
|
2019-07-13 17:50:23 +00:00
|
|
|
queueItems[k] = queueItems[fNodeCount--];
|
|
|
|
Downheap(k);
|
|
|
|
}
|
|
|
|
void Insert(uint32_t a1, T a2)
|
|
|
|
{
|
2020-10-11 10:38:17 +00:00
|
|
|
assert(fNodeCount < kPQueueSize);
|
2019-07-13 17:50:23 +00:00
|
|
|
fNodeCount++;
|
2020-09-01 19:27:32 +00:00
|
|
|
queueItems[fNodeCount].TotalKills = a1;
|
|
|
|
queueItems[fNodeCount].Kills = a2;
|
2019-07-13 17:50:23 +00:00
|
|
|
Upheap();
|
|
|
|
}
|
|
|
|
T Remove(void)
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
T data = queueItems[1].Kills;
|
2019-07-13 17:50:23 +00:00
|
|
|
queueItems[1] = queueItems[fNodeCount--];
|
|
|
|
Downheap(1);
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
uint32_t LowestPriority(void)
|
|
|
|
{
|
2020-10-11 10:38:17 +00:00
|
|
|
assert(fNodeCount > 0);
|
2020-09-01 19:27:32 +00:00
|
|
|
return queueItems[1].TotalKills;
|
2019-07-13 17:50:23 +00:00
|
|
|
}
|
|
|
|
void Kill(std::function<bool(T)> pMatch)
|
|
|
|
{
|
|
|
|
for (unsigned int i = 1; i <= fNodeCount;)
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
if (pMatch(queueItems[i].Kills))
|
2019-07-13 17:50:23 +00:00
|
|
|
Delete(i);
|
|
|
|
else
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
2019-09-19 22:42:45 +00:00
|
|
|
};
|
|
|
|
|
2019-07-13 17:50:23 +00:00
|
|
|
template<typename T> class StdPriorityQueue : public PriorityQueue<T>
|
2019-09-19 22:42:45 +00:00
|
|
|
{
|
|
|
|
public:
|
2019-07-13 17:50:23 +00:00
|
|
|
std::multiset<queueItem<T>> stdQueue;
|
|
|
|
~StdPriorityQueue()
|
|
|
|
{
|
|
|
|
stdQueue.clear();
|
|
|
|
}
|
|
|
|
uint32_t Size(void) { return stdQueue.size(); };
|
|
|
|
void Clear(void)
|
|
|
|
{
|
|
|
|
stdQueue.clear();
|
|
|
|
}
|
|
|
|
void Insert(uint32_t nPriority, T data)
|
|
|
|
{
|
|
|
|
stdQueue.insert({ nPriority, data });
|
|
|
|
}
|
|
|
|
T Remove(void)
|
|
|
|
{
|
2020-10-11 10:38:17 +00:00
|
|
|
assert(stdQueue.size() > 0);
|
2020-09-01 19:27:32 +00:00
|
|
|
T data = stdQueue.begin()->Kills;
|
2019-07-13 17:50:23 +00:00
|
|
|
stdQueue.erase(stdQueue.begin());
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
uint32_t LowestPriority(void)
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
return stdQueue.begin()->TotalKills;
|
2019-07-13 17:50:23 +00:00
|
|
|
}
|
|
|
|
void Kill(std::function<bool(T)> pMatch)
|
|
|
|
{
|
|
|
|
for (auto i = stdQueue.begin(); i != stdQueue.end();)
|
|
|
|
{
|
2020-09-01 19:27:32 +00:00
|
|
|
if (pMatch(i->Kills))
|
2019-07-13 17:50:23 +00:00
|
|
|
i = stdQueue.erase(i);
|
|
|
|
else
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
2019-09-19 22:42:45 +00:00
|
|
|
};
|
2019-09-22 06:39:22 +00:00
|
|
|
|
|
|
|
END_BLD_NS
|