453 lines
11 KiB
C
453 lines
11 KiB
C
/*************************************************************************
|
|
*
|
|
* Copyright (C) 2018-2020 Ruilin Peng (Nick) <pymumu@gmail.com>.
|
|
*
|
|
* smartdns 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 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* smartdns 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/>.
|
|
*/
|
|
|
|
#include "dns_cache.h"
|
|
#include "stringutil.h"
|
|
#include "tlog.h"
|
|
#include <pthread.h>
|
|
|
|
#define DNS_CACHE_MAX_HITNUM 5000
|
|
#define DNS_CACHE_HITNUM_STEP 2
|
|
#define DNS_CACHE_HITNUM_STEP_MAX 6
|
|
|
|
struct dns_cache_head {
|
|
DECLARE_HASHTABLE(cache_hash, 10);
|
|
struct list_head cache_list;
|
|
struct list_head inactive_list;
|
|
atomic_t num;
|
|
int size;
|
|
int enable_inactive;
|
|
int inactive_list_expired;
|
|
pthread_mutex_t lock;
|
|
};
|
|
|
|
static struct dns_cache_head dns_cache_head;
|
|
|
|
int dns_cache_init(int size, int enable_inactive, int inactive_list_expired)
|
|
{
|
|
INIT_LIST_HEAD(&dns_cache_head.cache_list);
|
|
INIT_LIST_HEAD(&dns_cache_head.inactive_list);
|
|
hash_init(dns_cache_head.cache_hash);
|
|
atomic_set(&dns_cache_head.num, 0);
|
|
dns_cache_head.size = size;
|
|
dns_cache_head.enable_inactive = enable_inactive;
|
|
dns_cache_head.inactive_list_expired = inactive_list_expired;
|
|
|
|
pthread_mutex_init(&dns_cache_head.lock, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static __attribute__((unused)) struct dns_cache *_dns_cache_last(void)
|
|
{
|
|
struct dns_cache *dns_cache = NULL;
|
|
|
|
dns_cache = list_last_entry(&dns_cache_head.inactive_list, struct dns_cache, list);
|
|
if (dns_cache) {
|
|
return dns_cache;
|
|
}
|
|
|
|
return list_last_entry(&dns_cache_head.cache_list, struct dns_cache, list);
|
|
}
|
|
|
|
static struct dns_cache *_dns_cache_first(void)
|
|
{
|
|
struct dns_cache *dns_cache = NULL;
|
|
|
|
dns_cache = list_first_entry_or_null(&dns_cache_head.inactive_list, struct dns_cache, list);
|
|
if (dns_cache) {
|
|
return dns_cache;
|
|
}
|
|
|
|
return list_first_entry_or_null(&dns_cache_head.cache_list, struct dns_cache, list);
|
|
}
|
|
|
|
static void _dns_cache_delete(struct dns_cache *dns_cache)
|
|
{
|
|
hash_del(&dns_cache->node);
|
|
list_del_init(&dns_cache->list);
|
|
atomic_dec(&dns_cache_head.num);
|
|
free(dns_cache);
|
|
}
|
|
|
|
void dns_cache_get(struct dns_cache *dns_cache)
|
|
{
|
|
if (atomic_inc_return(&dns_cache->ref) == 1) {
|
|
tlog(TLOG_ERROR, "BUG: dns_cache is invalid.");
|
|
return;
|
|
}
|
|
}
|
|
|
|
void dns_cache_release(struct dns_cache *dns_cache)
|
|
{
|
|
if (dns_cache == NULL) {
|
|
return;
|
|
}
|
|
if (!atomic_dec_and_test(&dns_cache->ref)) {
|
|
return;
|
|
}
|
|
|
|
_dns_cache_delete(dns_cache);
|
|
}
|
|
|
|
static void _dns_cache_remove(struct dns_cache *dns_cache)
|
|
{
|
|
hash_del(&dns_cache->node);
|
|
list_del_init(&dns_cache->list);
|
|
dns_cache_release(dns_cache);
|
|
}
|
|
|
|
static void _dns_cache_move_inactive(struct dns_cache *dns_cache)
|
|
{
|
|
list_del_init(&dns_cache->list);
|
|
list_add_tail(&dns_cache->list, &dns_cache_head.inactive_list);
|
|
}
|
|
|
|
int dns_cache_replace(char *domain, char *cname, int cname_ttl, int ttl, dns_type_t qtype, unsigned char *addr,
|
|
int addr_len, int speed)
|
|
{
|
|
struct dns_cache *dns_cache = NULL;
|
|
|
|
if (dns_cache_head.size <= 0) {
|
|
return 0;
|
|
}
|
|
|
|
/* lookup existing cache */
|
|
dns_cache = dns_cache_lookup(domain, qtype);
|
|
if (dns_cache == NULL) {
|
|
return dns_cache_insert(domain, cname, cname_ttl, ttl, qtype, addr, addr_len, speed);
|
|
}
|
|
|
|
if (ttl < DNS_CACHE_TTL_MIN) {
|
|
ttl = DNS_CACHE_TTL_MIN;
|
|
}
|
|
|
|
/* update cache data */
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
dns_cache->ttl = ttl;
|
|
dns_cache->qtype = qtype;
|
|
dns_cache->ttl = ttl;
|
|
dns_cache->del_pending = 0;
|
|
dns_cache->speed = speed;
|
|
time(&dns_cache->insert_time);
|
|
if (qtype == DNS_T_A) {
|
|
if (addr_len != DNS_RR_A_LEN) {
|
|
goto errout_unlock;
|
|
}
|
|
memcpy(dns_cache->addr, addr, DNS_RR_A_LEN);
|
|
} else if (qtype == DNS_T_AAAA) {
|
|
if (addr_len != DNS_RR_AAAA_LEN) {
|
|
goto errout_unlock;
|
|
}
|
|
memcpy(dns_cache->addr, addr, DNS_RR_AAAA_LEN);
|
|
} else {
|
|
goto errout_unlock;
|
|
}
|
|
|
|
if (cname) {
|
|
safe_strncpy(dns_cache->cname, cname, DNS_MAX_CNAME_LEN);
|
|
dns_cache->cname_ttl = cname_ttl;
|
|
}
|
|
|
|
list_del_init(&dns_cache->list);
|
|
list_add_tail(&dns_cache->list, &dns_cache_head.cache_list);
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
|
|
dns_cache_release(dns_cache);
|
|
return 0;
|
|
errout_unlock:
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
// errout:
|
|
if (dns_cache) {
|
|
dns_cache_release(dns_cache);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int dns_cache_insert(char *domain, char *cname, int cname_ttl, int ttl, dns_type_t qtype, unsigned char *addr,
|
|
int addr_len, int speed)
|
|
{
|
|
uint32_t key = 0;
|
|
struct dns_cache *dns_cache = NULL;
|
|
|
|
if (dns_cache_head.size <= 0) {
|
|
return 0;
|
|
}
|
|
|
|
/* if cache already exists, free */
|
|
dns_cache = dns_cache_lookup(domain, qtype);
|
|
if (dns_cache) {
|
|
dns_cache_delete(dns_cache);
|
|
dns_cache_release(dns_cache);
|
|
dns_cache = NULL;
|
|
}
|
|
|
|
dns_cache = malloc(sizeof(*dns_cache));
|
|
if (dns_cache == NULL) {
|
|
goto errout;
|
|
}
|
|
|
|
if (ttl < DNS_CACHE_TTL_MIN) {
|
|
ttl = DNS_CACHE_TTL_MIN;
|
|
}
|
|
|
|
key = hash_string(domain);
|
|
key = jhash(&qtype, sizeof(qtype), key);
|
|
safe_strncpy(dns_cache->domain, domain, DNS_MAX_CNAME_LEN);
|
|
dns_cache->cname[0] = 0;
|
|
dns_cache->qtype = qtype;
|
|
dns_cache->ttl = ttl;
|
|
atomic_set(&dns_cache->hitnum, 3);
|
|
dns_cache->hitnum_update_add = DNS_CACHE_HITNUM_STEP;
|
|
dns_cache->del_pending = 0;
|
|
dns_cache->speed = speed;
|
|
atomic_set(&dns_cache->ref, 1);
|
|
time(&dns_cache->insert_time);
|
|
if (qtype == DNS_T_A) {
|
|
if (addr_len != DNS_RR_A_LEN) {
|
|
goto errout;
|
|
}
|
|
memcpy(dns_cache->addr, addr, DNS_RR_A_LEN);
|
|
} else if (qtype == DNS_T_AAAA) {
|
|
if (addr_len != DNS_RR_AAAA_LEN) {
|
|
goto errout;
|
|
}
|
|
memcpy(dns_cache->addr, addr, DNS_RR_AAAA_LEN);
|
|
} else {
|
|
goto errout;
|
|
}
|
|
|
|
if (cname) {
|
|
safe_strncpy(dns_cache->cname, cname, DNS_MAX_CNAME_LEN);
|
|
dns_cache->cname_ttl = cname_ttl;
|
|
}
|
|
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
hash_add(dns_cache_head.cache_hash, &dns_cache->node, key);
|
|
list_add_tail(&dns_cache->list, &dns_cache_head.cache_list);
|
|
INIT_LIST_HEAD(&dns_cache->check_list);
|
|
|
|
/* Release extra cache, remove oldest cache record */
|
|
if (atomic_inc_return(&dns_cache_head.num) > dns_cache_head.size) {
|
|
struct dns_cache *del_cache;
|
|
del_cache = _dns_cache_first();
|
|
if (del_cache) {
|
|
_dns_cache_remove(del_cache);
|
|
}
|
|
}
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
|
|
return 0;
|
|
errout:
|
|
if (dns_cache) {
|
|
free(dns_cache);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
struct dns_cache *dns_cache_lookup(char *domain, dns_type_t qtype)
|
|
{
|
|
uint32_t key = 0;
|
|
struct dns_cache *dns_cache = NULL;
|
|
struct dns_cache *dns_cache_ret = NULL;
|
|
time_t now;
|
|
|
|
if (dns_cache_head.size <= 0) {
|
|
return NULL;
|
|
}
|
|
|
|
key = hash_string(domain);
|
|
key = jhash(&qtype, sizeof(qtype), key);
|
|
|
|
time(&now);
|
|
/* find cache */
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
hash_for_each_possible(dns_cache_head.cache_hash, dns_cache, node, key)
|
|
{
|
|
if (dns_cache->qtype != qtype) {
|
|
continue;
|
|
}
|
|
|
|
if (strncmp(domain, dns_cache->domain, DNS_MAX_CNAME_LEN) != 0) {
|
|
continue;
|
|
}
|
|
|
|
dns_cache_ret = dns_cache;
|
|
break;
|
|
}
|
|
|
|
if (dns_cache_ret) {
|
|
/* Return NULL if the cache times out */
|
|
if (dns_cache_head.enable_inactive == 0 && (now - dns_cache_ret->insert_time > dns_cache_ret->ttl)) {
|
|
_dns_cache_remove(dns_cache_ret);
|
|
dns_cache_ret = NULL;
|
|
} else {
|
|
dns_cache_get(dns_cache_ret);
|
|
}
|
|
}
|
|
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
|
|
return dns_cache_ret;
|
|
}
|
|
|
|
int dns_cache_get_ttl(struct dns_cache *dns_cache)
|
|
{
|
|
time_t now;
|
|
int ttl = 0;
|
|
time(&now);
|
|
|
|
ttl = dns_cache->insert_time + dns_cache->ttl - now;
|
|
if (ttl < 0) {
|
|
return 0;
|
|
}
|
|
|
|
return ttl;
|
|
}
|
|
|
|
void dns_cache_delete(struct dns_cache *dns_cache)
|
|
{
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
_dns_cache_remove(dns_cache);
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
}
|
|
|
|
int dns_cache_hitnum_dec_get(struct dns_cache *dns_cache)
|
|
{
|
|
int hitnum = 0;
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
hitnum = atomic_dec_return(&dns_cache->hitnum);
|
|
if (dns_cache->hitnum_update_add > DNS_CACHE_HITNUM_STEP) {
|
|
dns_cache->hitnum_update_add--;
|
|
}
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
|
|
return hitnum;
|
|
}
|
|
|
|
void dns_cache_update(struct dns_cache *dns_cache)
|
|
{
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
if (!list_empty(&dns_cache->list)) {
|
|
list_del_init(&dns_cache->list);
|
|
list_add_tail(&dns_cache->list, &dns_cache_head.cache_list);
|
|
atomic_add(dns_cache->hitnum_update_add, &dns_cache->hitnum);
|
|
if (atomic_read(&dns_cache->hitnum) > DNS_CACHE_MAX_HITNUM) {
|
|
atomic_set(&dns_cache->hitnum, DNS_CACHE_MAX_HITNUM);
|
|
}
|
|
|
|
if (dns_cache->hitnum_update_add < DNS_CACHE_HITNUM_STEP_MAX) {
|
|
dns_cache->hitnum_update_add++;
|
|
}
|
|
}
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
}
|
|
|
|
void _dns_cache_remove_expired_ttl(time_t *now)
|
|
{
|
|
struct dns_cache *dns_cache = NULL;
|
|
struct dns_cache *tmp;
|
|
int ttl = 0;
|
|
|
|
list_for_each_entry_safe(dns_cache, tmp, &dns_cache_head.inactive_list, list)
|
|
{
|
|
ttl = dns_cache->insert_time + dns_cache->ttl - *now;
|
|
if (ttl > 0) {
|
|
continue;
|
|
}
|
|
|
|
if (dns_cache_head.inactive_list_expired + ttl > 0) {
|
|
continue;
|
|
}
|
|
|
|
_dns_cache_remove(dns_cache);
|
|
}
|
|
}
|
|
|
|
void dns_cache_invalidate(dns_cache_preinvalid_callback callback, int ttl_pre)
|
|
{
|
|
struct dns_cache *dns_cache = NULL;
|
|
struct dns_cache *tmp;
|
|
time_t now;
|
|
int ttl = 0;
|
|
LIST_HEAD(checklist);
|
|
|
|
if (dns_cache_head.size <= 0) {
|
|
return;
|
|
}
|
|
|
|
time(&now);
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
list_for_each_entry_safe(dns_cache, tmp, &dns_cache_head.cache_list, list)
|
|
{
|
|
ttl = dns_cache->insert_time + dns_cache->ttl - now;
|
|
if (ttl > 0 && ttl < ttl_pre) {
|
|
/* If the TTL time is in the pre-timeout range, call callback function */
|
|
if (callback && dns_cache->del_pending == 0) {
|
|
list_add_tail(&dns_cache->check_list, &checklist);
|
|
dns_cache_get(dns_cache);
|
|
dns_cache->del_pending = 1;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (ttl < 0) {
|
|
if (dns_cache_head.enable_inactive) {
|
|
_dns_cache_move_inactive(dns_cache);
|
|
} else {
|
|
_dns_cache_remove(dns_cache);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (dns_cache_head.enable_inactive && dns_cache_head.inactive_list_expired != 0) {
|
|
_dns_cache_remove_expired_ttl(&now);
|
|
}
|
|
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
|
|
list_for_each_entry_safe(dns_cache, tmp, &checklist, check_list)
|
|
{
|
|
/* run callback */
|
|
if (callback) {
|
|
callback(dns_cache);
|
|
}
|
|
dns_cache_release(dns_cache);
|
|
}
|
|
}
|
|
|
|
void dns_cache_destroy(void)
|
|
{
|
|
struct dns_cache *dns_cache = NULL;
|
|
struct dns_cache *tmp;
|
|
pthread_mutex_lock(&dns_cache_head.lock);
|
|
list_for_each_entry_safe(dns_cache, tmp, &dns_cache_head.inactive_list, list)
|
|
{
|
|
_dns_cache_delete(dns_cache);
|
|
}
|
|
|
|
list_for_each_entry_safe(dns_cache, tmp, &dns_cache_head.cache_list, list)
|
|
{
|
|
_dns_cache_delete(dns_cache);
|
|
}
|
|
pthread_mutex_unlock(&dns_cache_head.lock);
|
|
|
|
pthread_mutex_destroy(&dns_cache_head.lock);
|
|
}
|