176 lines
5.3 KiB
C++
176 lines
5.3 KiB
C++
/**
|
|
* @file aiperhost.cpp
|
|
* @brief Implementation of PerHostRequestQueue
|
|
*
|
|
* Copyright (c) 2012, Aleric Inglewood.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* There are special exceptions to the terms and conditions of the GPL as
|
|
* it is applied to this Source Code. View the full text of the exception
|
|
* in the file doc/FLOSS-exception.txt in this software distribution.
|
|
*
|
|
* CHANGELOG
|
|
* and additional copyright holders.
|
|
*
|
|
* 04/11/2012
|
|
* Initial version, written by Aleric Inglewood @ SL
|
|
*/
|
|
|
|
#include "sys.h"
|
|
#include "aicurlperhost.h"
|
|
#include "aicurlthread.h"
|
|
|
|
#undef AICurlPrivate
|
|
|
|
namespace AICurlPrivate {
|
|
|
|
PerHostRequestQueue::threadsafe_instance_map_type PerHostRequestQueue::sInstanceMap;
|
|
U32 curl_concurrent_connections_per_host;
|
|
|
|
//static
|
|
PerHostRequestQueuePtr PerHostRequestQueue::instance(std::string const& hostname)
|
|
{
|
|
llassert(!hostname.empty());
|
|
instance_map_wat instance_map_w(sInstanceMap);
|
|
PerHostRequestQueue::iterator iter = instance_map_w->find(hostname);
|
|
if (iter == instance_map_w->end())
|
|
{
|
|
iter = instance_map_w->insert(instance_map_type::value_type(hostname, new RefCountedThreadSafePerHostRequestQueue)).first;
|
|
}
|
|
// Note: the creation of PerHostRequestQueuePtr MUST be protected by the lock on sInstanceMap (see release()).
|
|
return iter->second;
|
|
}
|
|
|
|
//static
|
|
void PerHostRequestQueue::release(PerHostRequestQueuePtr& instance)
|
|
{
|
|
if (instance->exactly_two_left()) // Being 'instance' and the one in sInstanceMap.
|
|
{
|
|
// The viewer can be have left main() we can't access the global sInstanceMap anymore.
|
|
if (LLApp::isStopped())
|
|
{
|
|
return;
|
|
}
|
|
instance_map_wat instance_map_w(sInstanceMap);
|
|
// It is possible that 'exactly_two_left' is not up to date anymore.
|
|
// Therefore, recheck the condition now that we have locked sInstanceMap.
|
|
if (!instance->exactly_two_left())
|
|
{
|
|
// Some other thread added this host in the meantime.
|
|
return;
|
|
}
|
|
// The reference in the map is the last one; that means there can't be any curl easy requests queued for this host.
|
|
llassert(PerHostRequestQueue_wat(*instance)->mQueuedRequests.empty());
|
|
// Find the host and erase it from the map.
|
|
iterator const end = instance_map_w->end();
|
|
for(iterator iter = instance_map_w->begin(); iter != end; ++iter)
|
|
{
|
|
if (instance == iter->second)
|
|
{
|
|
instance_map_w->erase(iter);
|
|
instance.reset();
|
|
return;
|
|
}
|
|
}
|
|
// We should always find the host.
|
|
llassert(false);
|
|
}
|
|
instance.reset();
|
|
}
|
|
|
|
bool PerHostRequestQueue::throttled() const
|
|
{
|
|
llassert(mAdded <= curl_concurrent_connections_per_host);
|
|
return mAdded == curl_concurrent_connections_per_host;
|
|
}
|
|
|
|
void PerHostRequestQueue::added_to_multi_handle(void)
|
|
{
|
|
llassert(mAdded < curl_concurrent_connections_per_host);
|
|
++mAdded;
|
|
}
|
|
|
|
void PerHostRequestQueue::removed_from_multi_handle(void)
|
|
{
|
|
--mAdded;
|
|
llassert(mAdded >= 0);
|
|
}
|
|
|
|
void PerHostRequestQueue::queue(AICurlEasyRequest const& easy_request)
|
|
{
|
|
mQueuedRequests.push_back(easy_request.get_ptr());
|
|
}
|
|
|
|
bool PerHostRequestQueue::cancel(AICurlEasyRequest const& easy_request)
|
|
{
|
|
queued_request_type::iterator const end = mQueuedRequests.end();
|
|
queued_request_type::iterator cur = std::find(mQueuedRequests.begin(), end, easy_request.get_ptr());
|
|
|
|
if (cur == end)
|
|
return false; // Not found.
|
|
|
|
// We can't use erase because that uses assignment to move elements,
|
|
// because it isn't thread-safe. Therefore, move the element that we found to
|
|
// the back with swap (could just swap with the end immediately, but I don't
|
|
// want to break the order in which requests where added). Swap is also not
|
|
// thread-safe, but OK here because it only touches the objects in the deque,
|
|
// and the deque is protected by the lock on the PerHostRequestQueue object.
|
|
queued_request_type::iterator prev = cur;
|
|
while (++cur != end)
|
|
{
|
|
prev->swap(*cur); // This is safe,
|
|
prev = cur;
|
|
}
|
|
mQueuedRequests.pop_back(); // if this is safe.
|
|
return true;
|
|
}
|
|
|
|
void PerHostRequestQueue::add_queued_to(curlthread::MultiHandle* multi_handle)
|
|
{
|
|
if (!mQueuedRequests.empty())
|
|
{
|
|
multi_handle->add_easy_request(mQueuedRequests.front());
|
|
mQueuedRequests.pop_front();
|
|
}
|
|
}
|
|
|
|
//static
|
|
void PerHostRequestQueue::purge(void)
|
|
{
|
|
instance_map_wat instance_map_w(sInstanceMap);
|
|
for (iterator host = instance_map_w->begin(); host != instance_map_w->end(); ++host)
|
|
{
|
|
Dout(dc::curl, "Purging queue of host \"" << host->first << "\".");
|
|
PerHostRequestQueue_wat(*host->second)->mQueuedRequests.clear();
|
|
}
|
|
}
|
|
|
|
// Friend functions of RefCountedThreadSafePerHostRequestQueue
|
|
|
|
void intrusive_ptr_add_ref(RefCountedThreadSafePerHostRequestQueue* per_host)
|
|
{
|
|
per_host->mReferenceCount++;
|
|
}
|
|
|
|
void intrusive_ptr_release(RefCountedThreadSafePerHostRequestQueue* per_host)
|
|
{
|
|
if (--per_host->mReferenceCount == 0)
|
|
{
|
|
delete per_host;
|
|
}
|
|
}
|
|
|
|
} // namespace AICurlPrivate
|