1
0
mirror of git://erdgeist.org/opentracker synced 2025-01-13 00:10:08 +00:00
opentracker/ot_clean.c

145 lines
4.3 KiB
C
Raw Normal View History

/* This software was written by Dirk Engling <erdgeist@erdgeist.org>
2007-12-20 05:59:34 +00:00
It is considered beerware. Prost. Skol. Cheers or whatever.
2008-10-28 01:27:22 +00:00
2007-12-20 05:59:34 +00:00
$id$ */
/* System */
#include <pthread.h>
#include <string.h>
2024-04-14 22:39:02 +00:00
#include <unistd.h>
/* Libowfat */
2007-12-03 00:52:05 +00:00
#include "io.h"
/* Opentracker */
2024-04-14 22:39:02 +00:00
#include "ot_accesslist.h"
#include "ot_clean.h"
2024-04-14 22:39:02 +00:00
#include "ot_mutex.h"
#include "ot_stats.h"
2024-04-14 22:39:02 +00:00
#include "ot_vector.h"
#include "trackerlogic.h"
/* Returns amount of removed peers */
2024-04-14 22:39:02 +00:00
static ssize_t clean_single_bucket(ot_peer *peers, size_t peer_count, size_t peer_size, time_t timedout, int *removed_seeders) {
ot_peer *last_peer = peers + peer_count * peer_size, *insert_point;
/* Two scan modes: unless there is one peer removed, just increase ot_peertime */
2024-04-14 22:39:02 +00:00
while (peers < last_peer) {
time_t timediff = timedout + OT_PEERTIME(peers, peer_size);
if (timediff >= OT_PEER_TIMEOUT)
break;
2024-04-14 22:39:02 +00:00
OT_PEERTIME(peers, peer_size) = timediff;
peers += peer_size;
}
/* If we at least remove one peer, we have to copy */
2024-04-14 22:39:02 +00:00
for (insert_point = peers; peers < last_peer; peers += peer_size) {
time_t timediff = timedout + OT_PEERTIME(peers, peer_size);
2024-04-14 22:39:02 +00:00
if (timediff < OT_PEER_TIMEOUT) {
OT_PEERTIME(peers, peer_size) = timediff;
memcpy(insert_point, peers, peer_size);
insert_point += peer_size;
2024-04-14 22:39:02 +00:00
} else if (OT_PEERFLAG_D(peers, peer_size) & PEER_FLAG_SEEDING)
(*removed_seeders)++;
}
return (peers - insert_point) / peer_size;
}
2024-04-14 22:39:02 +00:00
int clean_single_peer_list(ot_peerlist *peer_list, size_t peer_size) {
ot_vector *peer_vector = &peer_list->peers;
2024-04-14 22:39:02 +00:00
time_t timedout = (time_t)(g_now_minutes - peer_list->base);
int num_buckets = 1, removed_seeders = 0;
/* No need to clean empty torrent */
2024-04-14 22:39:02 +00:00
if (!timedout)
return 0;
/* Torrent has idled out */
2024-04-14 22:39:02 +00:00
if (timedout > OT_TORRENT_TIMEOUT)
return 1;
/* Nothing to be cleaned here? Test if torrent is worth keeping */
2024-04-14 22:39:02 +00:00
if (timedout > OT_PEER_TIMEOUT) {
if (!peer_list->peer_count)
return peer_list->down_count ? 0 : 1;
timedout = OT_PEER_TIMEOUT;
}
2024-04-14 22:39:02 +00:00
if (OT_PEERLIST_HASBUCKETS(peer_list)) {
num_buckets = peer_vector->size;
peer_vector = (ot_vector *)peer_vector->data;
}
2024-04-14 22:39:02 +00:00
while (num_buckets--) {
size_t removed_peers = clean_single_bucket(peer_vector->data, peer_vector->size, peer_size, timedout, &removed_seeders);
peer_list->peer_count -= removed_peers;
peer_vector->size -= removed_peers;
2024-04-14 22:39:02 +00:00
if (removed_peers)
vector_fixup_peers(peer_vector, peer_size);
/* Skip to next bucket, a vector containing peers */
++peer_vector;
}
peer_list->seed_count -= removed_seeders;
/* See if we need to convert a torrent from simple vector to bucket list */
2024-04-14 22:39:02 +00:00
if ((peer_list->peer_count > OT_PEER_BUCKET_MINCOUNT) || OT_PEERLIST_HASBUCKETS(peer_list))
vector_redistribute_buckets(peer_list, peer_size);
2024-04-14 22:39:02 +00:00
if (peer_list->peer_count)
peer_list->base = g_now_minutes;
else {
/* When we got here, the last time that torrent
has been touched is OT_PEER_TIMEOUT Minutes before */
peer_list->base = g_now_minutes - OT_PEER_TIMEOUT;
}
return 0;
}
/* Clean a single torrent
return 1 if torrent timed out
*/
2024-04-14 22:39:02 +00:00
int clean_single_torrent(ot_torrent *torrent) {
return clean_single_peer_list(torrent->peer_list6, OT_PEER_SIZE6) * clean_single_peer_list(torrent->peer_list4, OT_PEER_SIZE4);
}
/* Clean up all peers in current bucket, remove timedout pools and
torrents */
2024-04-14 22:39:02 +00:00
static void *clean_worker(void *args) {
(void)args;
while (1) {
int bucket = OT_BUCKET_COUNT;
2024-04-14 22:39:02 +00:00
while (bucket--) {
ot_vector *torrents_list = mutex_bucket_lock(bucket);
size_t toffs;
int delta_torrentcount = 0;
2024-04-14 22:39:02 +00:00
for (toffs = 0; toffs < torrents_list->size; ++toffs) {
ot_torrent *torrent = ((ot_torrent *)(torrents_list->data)) + toffs;
if (clean_single_torrent(torrent)) {
vector_remove_torrent(torrents_list, torrent);
2009-02-20 12:23:59 +00:00
--delta_torrentcount;
--toffs;
}
}
2024-04-14 22:39:02 +00:00
mutex_bucket_unlock(bucket, delta_torrentcount);
if (!g_opentracker_running)
return NULL;
2024-04-14 22:39:02 +00:00
usleep(OT_CLEAN_SLEEP);
}
stats_cleanup();
2022-11-24 15:38:26 +00:00
#ifdef WANT_ACCESSLIST
accesslist_cleanup();
#endif
}
return NULL;
}
static pthread_t thread_id;
2024-04-14 22:39:02 +00:00
void clean_init(void) { pthread_create(&thread_id, NULL, clean_worker, NULL); }
2024-04-14 22:39:02 +00:00
void clean_deinit(void) { pthread_cancel(thread_id); }