Merge pull request #404 from rockuw/keep-alive

Add curl handler pool to reuse connections
This commit is contained in:
Takeshi Nakatani 2016-04-26 23:40:45 +09:00
commit 845fdb43f2
2 changed files with 126 additions and 2 deletions

View File

@ -220,6 +220,88 @@ const char* BodyData::str(void) const
return text; return text;
} }
//-------------------------------------------------------------------
// Class CurlHandlerPool
//-------------------------------------------------------------------
bool CurlHandlerPool::Init()
{
if (0 != pthread_mutex_init(&mLock, NULL)) {
S3FS_PRN_ERR("Init curl handlers lock failed");
return false;
}
mHandlers = new CURL*[mMaxHandlers](); // this will init the array to 0
for (int i = 0; i < mMaxHandlers; ++i, ++mIndex) {
mHandlers[i] = curl_easy_init();
if (!mHandlers[i]) {
S3FS_PRN_ERR("Init curl handlers pool failed");
Destroy();
return false;
}
}
return true;
}
bool CurlHandlerPool::Destroy()
{
assert(mIndex >= -1 && mIndex < mMaxHandlers);
for (int i = 0; i <= mIndex; ++i) {
curl_easy_cleanup(mHandlers[i]);
}
delete[] mHandlers;
if (0 != pthread_mutex_destroy(&mLock)) {
S3FS_PRN_ERR("Destroy curl handlers lock failed");
return false;
}
return true;
}
CURL* CurlHandlerPool::GetHandler()
{
CURL* h = NULL;
assert(mIndex >= -1 && mIndex < mMaxHandlers);
pthread_mutex_lock(&mLock);
if (mIndex >= 0) {
S3FS_PRN_DBG("Get handler from pool: %d", mIndex);
h = mHandlers[mIndex--];
}
pthread_mutex_unlock(&mLock);
if (!h) {
S3FS_PRN_INFO("Pool empty: create new handler");
h = curl_easy_init();
}
return h;
}
void CurlHandlerPool::ReturnHandler(CURL* h)
{
bool needCleanup = true;
assert(mIndex >= -1 && mIndex < mMaxHandlers);
pthread_mutex_lock(&mLock);
if (mIndex < mMaxHandlers - 1) {
mHandlers[++mIndex] = h;
needCleanup = false;
S3FS_PRN_DBG("Return handler to pool: %d", mIndex);
}
pthread_mutex_unlock(&mLock);
if (needCleanup) {
S3FS_PRN_INFO("Pool full: destroy the handler");
curl_easy_cleanup(h);
}
}
//------------------------------------------------------------------- //-------------------------------------------------------------------
// Class S3fsCurl // Class S3fsCurl
//------------------------------------------------------------------- //-------------------------------------------------------------------
@ -243,6 +325,8 @@ const char* BodyData::str(void) const
pthread_mutex_t S3fsCurl::curl_handles_lock; pthread_mutex_t S3fsCurl::curl_handles_lock;
pthread_mutex_t S3fsCurl::curl_share_lock[SHARE_MUTEX_MAX]; pthread_mutex_t S3fsCurl::curl_share_lock[SHARE_MUTEX_MAX];
bool S3fsCurl::is_initglobal_done = false; bool S3fsCurl::is_initglobal_done = false;
CurlHandlerPool* S3fsCurl::sCurlPool = NULL;
int S3fsCurl::sCurlPoolSize = 32;
CURLSH* S3fsCurl::hCurlShare = NULL; CURLSH* S3fsCurl::hCurlShare = NULL;
bool S3fsCurl::is_cert_check = true; // default bool S3fsCurl::is_cert_check = true; // default
bool S3fsCurl::is_dns_cache = true; // default bool S3fsCurl::is_dns_cache = true; // default
@ -293,6 +377,10 @@ bool S3fsCurl::InitS3fsCurl(const char* MimeFile)
if(!S3fsCurl::InitGlobalCurl()){ if(!S3fsCurl::InitGlobalCurl()){
return false; return false;
} }
sCurlPool = new CurlHandlerPool(sCurlPoolSize);
if (!sCurlPool->Init()) {
return false;
}
if(!S3fsCurl::InitShareCurl()){ if(!S3fsCurl::InitShareCurl()){
return false; return false;
} }
@ -312,6 +400,9 @@ bool S3fsCurl::DestroyS3fsCurl(void)
if(!S3fsCurl::DestroyShareCurl()){ if(!S3fsCurl::DestroyShareCurl()){
result = false; result = false;
} }
if (!sCurlPool->Destroy()) {
result = false;
}
if(!S3fsCurl::DestroyGlobalCurl()){ if(!S3fsCurl::DestroyGlobalCurl()){
result = false; result = false;
} }
@ -1478,7 +1569,7 @@ bool S3fsCurl::CreateCurlHandle(bool force)
S3FS_PRN_INFO3("already has handle, so destroied it."); S3FS_PRN_INFO3("already has handle, so destroied it.");
} }
if(NULL == (hCurl = curl_easy_init())){ if(NULL == (hCurl = sCurlPool->GetHandler())){
S3FS_PRN_ERR("Failed to create handle."); S3FS_PRN_ERR("Failed to create handle.");
return false; return false;
} }
@ -1508,7 +1599,7 @@ bool S3fsCurl::DestroyCurlHandle(void)
S3fsCurl::curl_times.erase(hCurl); S3fsCurl::curl_times.erase(hCurl);
S3fsCurl::curl_progress.erase(hCurl); S3fsCurl::curl_progress.erase(hCurl);
curl_easy_cleanup(hCurl); sCurlPool->ReturnHandler(hCurl);
hCurl = NULL; hCurl = NULL;
ClearInternalData(); ClearInternalData();

View File

@ -21,6 +21,8 @@
#ifndef S3FS_CURL_H_ #ifndef S3FS_CURL_H_
#define S3FS_CURL_H_ #define S3FS_CURL_H_
#include <cassert>
//---------------------------------------------- //----------------------------------------------
// Symbols // Symbols
//---------------------------------------------- //----------------------------------------------
@ -121,6 +123,35 @@ typedef std::map<CURL*, progress_t> curlprogress_t;
class S3fsMultiCurl; class S3fsMultiCurl;
//----------------------------------------------
// class CurlHandlerPool
//----------------------------------------------
class CurlHandlerPool
{
public:
CurlHandlerPool(int maxHandlers)
: mMaxHandlers(maxHandlers)
, mHandlers(NULL)
, mIndex(-1)
{
assert(maxHandlers > 0);
}
bool Init();
bool Destroy();
CURL* GetHandler();
void ReturnHandler(CURL* h);
private:
int mMaxHandlers;
pthread_mutex_t mLock;
CURL** mHandlers;
int mIndex;
};
//---------------------------------------------- //----------------------------------------------
// class S3fsCurl // class S3fsCurl
//---------------------------------------------- //----------------------------------------------
@ -177,6 +208,8 @@ class S3fsCurl
static pthread_mutex_t curl_handles_lock; static pthread_mutex_t curl_handles_lock;
static pthread_mutex_t curl_share_lock[SHARE_MUTEX_MAX]; static pthread_mutex_t curl_share_lock[SHARE_MUTEX_MAX];
static bool is_initglobal_done; static bool is_initglobal_done;
static CurlHandlerPool* sCurlPool;
static int sCurlPoolSize;
static CURLSH* hCurlShare; static CURLSH* hCurlShare;
static bool is_cert_check; static bool is_cert_check;
static bool is_dns_cache; static bool is_dns_cache;