Merge "Avoid multiple dns lookups for the same query"
This commit is contained in:
commit
028ccf5d40
@ -1159,6 +1159,15 @@ entry_equals( const Entry* e1, const Entry* e2 )
|
|||||||
* inlined in the Entry structure.
|
* inlined in the Entry structure.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/* Maximum time for a thread to wait for an pending request */
|
||||||
|
#define PENDING_REQUEST_TIMEOUT 20;
|
||||||
|
|
||||||
|
typedef struct pending_req_info {
|
||||||
|
unsigned int hash;
|
||||||
|
pthread_cond_t cond;
|
||||||
|
struct pending_req_info* next;
|
||||||
|
} PendingReqInfo;
|
||||||
|
|
||||||
typedef struct resolv_cache {
|
typedef struct resolv_cache {
|
||||||
int max_entries;
|
int max_entries;
|
||||||
int num_entries;
|
int num_entries;
|
||||||
@ -1167,6 +1176,7 @@ typedef struct resolv_cache {
|
|||||||
unsigned generation;
|
unsigned generation;
|
||||||
int last_id;
|
int last_id;
|
||||||
Entry* entries;
|
Entry* entries;
|
||||||
|
PendingReqInfo pending_requests;
|
||||||
} Cache;
|
} Cache;
|
||||||
|
|
||||||
typedef struct resolv_cache_info {
|
typedef struct resolv_cache_info {
|
||||||
@ -1180,6 +1190,107 @@ typedef struct resolv_cache_info {
|
|||||||
|
|
||||||
#define HTABLE_VALID(x) ((x) != NULL && (x) != HTABLE_DELETED)
|
#define HTABLE_VALID(x) ((x) != NULL && (x) != HTABLE_DELETED)
|
||||||
|
|
||||||
|
static void
|
||||||
|
_cache_flush_pending_requests_locked( struct resolv_cache* cache )
|
||||||
|
{
|
||||||
|
struct pending_req_info *ri, *tmp;
|
||||||
|
if (cache) {
|
||||||
|
ri = cache->pending_requests.next;
|
||||||
|
|
||||||
|
while (ri) {
|
||||||
|
tmp = ri;
|
||||||
|
ri = ri->next;
|
||||||
|
pthread_cond_broadcast(&tmp->cond);
|
||||||
|
|
||||||
|
pthread_cond_destroy(&tmp->cond);
|
||||||
|
free(tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
cache->pending_requests.next = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* return 0 if no pending request is found matching the key
|
||||||
|
* if a matching request is found the calling thread will wait
|
||||||
|
* and return 1 when released */
|
||||||
|
static int
|
||||||
|
_cache_check_pending_request_locked( struct resolv_cache* cache, Entry* key )
|
||||||
|
{
|
||||||
|
struct pending_req_info *ri, *prev;
|
||||||
|
int exist = 0;
|
||||||
|
|
||||||
|
if (cache && key) {
|
||||||
|
ri = cache->pending_requests.next;
|
||||||
|
prev = &cache->pending_requests;
|
||||||
|
while (ri) {
|
||||||
|
if (ri->hash == key->hash) {
|
||||||
|
exist = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
prev = ri;
|
||||||
|
ri = ri->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!exist) {
|
||||||
|
ri = calloc(1, sizeof(struct pending_req_info));
|
||||||
|
if (ri) {
|
||||||
|
ri->hash = key->hash;
|
||||||
|
pthread_cond_init(&ri->cond, NULL);
|
||||||
|
prev->next = ri;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
struct timespec ts = {0,0};
|
||||||
|
ts.tv_sec = _time_now() + PENDING_REQUEST_TIMEOUT;
|
||||||
|
int rv = pthread_cond_timedwait(&ri->cond, &cache->lock, &ts);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return exist;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* notify any waiting thread that waiting on a request
|
||||||
|
* matching the key has been added to the cache */
|
||||||
|
static void
|
||||||
|
_cache_notify_waiting_tid_locked( struct resolv_cache* cache, Entry* key )
|
||||||
|
{
|
||||||
|
struct pending_req_info *ri, *prev;
|
||||||
|
|
||||||
|
if (cache && key) {
|
||||||
|
ri = cache->pending_requests.next;
|
||||||
|
prev = &cache->pending_requests;
|
||||||
|
while (ri) {
|
||||||
|
if (ri->hash == key->hash) {
|
||||||
|
pthread_cond_broadcast(&ri->cond);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
prev = ri;
|
||||||
|
ri = ri->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove item from list and destroy
|
||||||
|
if (ri) {
|
||||||
|
prev->next = ri->next;
|
||||||
|
pthread_cond_destroy(&ri->cond);
|
||||||
|
free(ri);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* notify the cache that the query failed */
|
||||||
|
void
|
||||||
|
_resolv_cache_query_failed( struct resolv_cache* cache,
|
||||||
|
const void* query,
|
||||||
|
int querylen)
|
||||||
|
{
|
||||||
|
Entry key[1];
|
||||||
|
|
||||||
|
if (cache && entry_init_key(key, query, querylen)) {
|
||||||
|
pthread_mutex_lock(&cache->lock);
|
||||||
|
_cache_notify_waiting_tid_locked(cache, key);
|
||||||
|
pthread_mutex_unlock(&cache->lock);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
_cache_flush_locked( Cache* cache )
|
_cache_flush_locked( Cache* cache )
|
||||||
{
|
{
|
||||||
@ -1197,6 +1308,9 @@ _cache_flush_locked( Cache* cache )
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// flush pending request
|
||||||
|
_cache_flush_pending_requests_locked(cache);
|
||||||
|
|
||||||
cache->mru_list.mru_next = cache->mru_list.mru_prev = &cache->mru_list;
|
cache->mru_list.mru_next = cache->mru_list.mru_prev = &cache->mru_list;
|
||||||
cache->num_entries = 0;
|
cache->num_entries = 0;
|
||||||
cache->last_id = 0;
|
cache->last_id = 0;
|
||||||
@ -1470,7 +1584,17 @@ _resolv_cache_lookup( struct resolv_cache* cache,
|
|||||||
|
|
||||||
if (e == NULL) {
|
if (e == NULL) {
|
||||||
XLOG( "NOT IN CACHE");
|
XLOG( "NOT IN CACHE");
|
||||||
goto Exit;
|
// calling thread will wait if an outstanding request is found
|
||||||
|
// that matching this query
|
||||||
|
if (!_cache_check_pending_request_locked(cache, key)) {
|
||||||
|
goto Exit;
|
||||||
|
} else {
|
||||||
|
lookup = _cache_lookup_p(cache, key);
|
||||||
|
e = *lookup;
|
||||||
|
if (e == NULL) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
now = _time_now();
|
now = _time_now();
|
||||||
@ -1573,6 +1697,7 @@ _resolv_cache_add( struct resolv_cache* cache,
|
|||||||
_cache_dump_mru(cache);
|
_cache_dump_mru(cache);
|
||||||
#endif
|
#endif
|
||||||
Exit:
|
Exit:
|
||||||
|
_cache_notify_waiting_tid_locked(cache, key);
|
||||||
pthread_mutex_unlock( &cache->lock );
|
pthread_mutex_unlock( &cache->lock );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -646,6 +646,9 @@ res_nsend(res_state statp,
|
|||||||
errno = terrno;
|
errno = terrno;
|
||||||
return (-1);
|
return (-1);
|
||||||
fail:
|
fail:
|
||||||
|
#if USE_RESOLV_CACHE
|
||||||
|
_resolv_cache_query_failed(cache, buf, buflen);
|
||||||
|
#endif
|
||||||
res_nclose(statp);
|
res_nclose(statp);
|
||||||
return (-1);
|
return (-1);
|
||||||
}
|
}
|
||||||
|
@ -95,4 +95,10 @@ _resolv_cache_add( struct resolv_cache* cache,
|
|||||||
const void* answer,
|
const void* answer,
|
||||||
int answerlen );
|
int answerlen );
|
||||||
|
|
||||||
|
/* Notify the cache a request failed */
|
||||||
|
extern void
|
||||||
|
_resolv_cache_query_failed( struct resolv_cache* cache,
|
||||||
|
const void* query,
|
||||||
|
int querylen);
|
||||||
|
|
||||||
#endif /* _RESOLV_CACHE_H_ */
|
#endif /* _RESOLV_CACHE_H_ */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user