aboutsummaryrefslogtreecommitdiff
path: root/slowloris.c
blob: 2dca2082ce31b9702bd9454ffaa5a49a558b7a53 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
#include <stdio.h>
#include <sys/socket.h>
#include <dlfcn.h>
#include <netinet/tcp.h>
#include <netinet/in.h>
#include <unistd.h>
#include <stdlib.h>

static int silent_mode = 0;
static unsigned send_delay = 0;
static unsigned send_chunk = 1;

char* xgetenv(const char* name, char* def){
	char* rv = getenv(name);
	return rv ? rv : def;
}

__attribute__((constructor)) void slowloris_init(){
	silent_mode = strtoul(xgetenv("SLOWLORIS_SILENT", "0"), NULL, 10);
	send_delay = strtoul(xgetenv("SLOWLORIS_DELAY", "0"), NULL, 10);
	send_chunk = strtoul(xgetenv("SLOWLORIS_CHUNK", "1"), NULL, 10);
	if(!silent_mode){
		printf("preload-slowloris active\nDelay: %d\nChunking: %d\n", send_delay, send_chunk);
	}
}

int socket(int domain, int type, int protocol){
	static int (*real_socket) (int, int, int) = NULL;
	if(!real_socket){
		real_socket = dlsym(RTLD_NEXT, "socket");
	}

	int fd = real_socket(domain, type, protocol);
	int enable = 1;

	if(domain == AF_INET && type == SOCK_STREAM){
		if(setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &enable, sizeof(enable)) < 0){
			if(!silent_mode){
				perror("setsockopt/tcp_nodelay");
			}
		}
	}
	return fd;
}

ssize_t send(int sockfd, const void* buf, size_t len, int flags){
	static ssize_t (*real_send) (int, void*, size_t, int) = NULL;
	if(!real_send){
		real_send = dlsym(RTLD_NEXT, "send");
	}

	ssize_t rv = 0;

	for(size_t u = 0; u < len; u += send_chunk){
		ssize_t sent = real_send(sockfd, buf + u, (send_chunk < len - u) ? send_chunk : (len - u), flags);
		if(sent < 0){
			return -1;
		}
		rv += sent;
		if(send_delay){
			usleep(send_delay);
		}
	}
	return rv;
}