| | |
| | | // Variables are only modified before threads are created and as such are |
| | | // thread-safe tests_done is modified to provide a rough count of number of |
| | | // tests being completed for progress measurement |
| | | int tests_done = 0, num_tests = NUM_TESTS, num_servers = NUM_DNS, num_hosts = 0; |
| | | int tests_done = 0, num_tests = NUM_TESTS, num_servers = 0, num_hosts = 0; |
| | | struct hosts_list* hosts = NULL; |
| | | struct dns_list* servers = NULL; |
| | | |
| | |
| | | printf( |
| | | "\t-t <number>\tspecify the number of hostnames to be tested, " |
| | | "capped at 50 + number manually added\n\t\t\tdefaults to test " |
| | | "all " |
| | | "all " |
| | | "available\n"); |
| | | printf( |
| | | "\t-n <number>\tspecify the number of tests to perform on each " |
| | |
| | | } |
| | | for (int i = 0; i < NUM_DNS; i++) { |
| | | add_dns_server(&servers, (char*)DNS_SERVERS[i]); |
| | | num_servers++; |
| | | } |
| | | test_dns(); |
| | | sort_servers(&servers); |
| | |
| | | { |
| | | struct dns_list* curr = servers; |
| | | int i = 0; |
| | | pthread_t* threads = malloc(num_servers * sizeof(pthread_t)); |
| | | int init_num_servers = num_servers; |
| | | pthread_t* threads; |
| | | pthread_t progress; |
| | | // Check each server for reachability, can't be done in parallel as incorrect packets are received |
| | | // reachable() requires raw packets, which needs root |
| | | if (getuid() == 0) { |
| | | while (curr) { |
| | | printf("\rTesting reachability: %d/%d", i, init_num_servers); fflush(stdout); |
| | | unsigned char* buf[65535]; |
| | | int error_count = 0; |
| | | curr->errors = 0; |
| | | // retry 10 times for UDP unreliability |
| | | for (int i = 0; i < 10; i++) { |
| | | error_count += (reachable((unsigned char*)buf, curr->server) != 0); |
| | | } |
| | | /* fprintf(stderr, "ip: %s, ec: %d\n", curr->server, error_count); */ |
| | | // 30% error rate means unreachable |
| | | if (error_count >= 3) { |
| | | curr->errors = -1; |
| | | num_servers--; |
| | | } |
| | | curr = curr->next; |
| | | i++; |
| | | } |
| | | printf("\rTesting reachability: %d/%d\n", i, init_num_servers); |
| | | } |
| | | |
| | | threads = malloc(num_servers * sizeof(pthread_t)); |
| | | curr = servers; |
| | | i = 0; |
| | | pthread_create(&progress, NULL, print_progress, NULL); |
| | | while (curr) { |
| | | pthread_create(&threads[i], NULL, test_server, (void*)curr); |
| | | if (curr->errors == 0) { |
| | | pthread_create(&threads[i], NULL, test_server, (void*)curr); |
| | | i++; |
| | | } |
| | | curr = curr->next; |
| | | i++; |
| | | } |
| | | for (int i = 0; i < num_servers; i++) { |
| | | pthread_join(threads[i], NULL); |
| | |
| | | struct dns_list* dns = (struct dns_list*)in; |
| | | dns->time.tv_sec = 0; |
| | | dns->time.tv_nsec = 0; |
| | | dns->errors = 0; |
| | | for (int i = 0; i < num_tests; i++) { |
| | | struct hosts_list* curr = hosts; |
| | | while (curr) { |
| | |
| | | for (int j = 0; j < 3 && run.tv_sec == -1; j++) { |
| | | run = resolve(buf, curr->server, dns->server, T_A); |
| | | } |
| | | if (run.tv_sec == -1) { // if test has failed 3 times, set time taken |
| | | // to 3s as penalty |
| | | dns->errors++; |
| | | } else { |
| | | dns->time.tv_sec += run.tv_sec; |
| | | dns->time.tv_nsec += run.tv_nsec; |
| | | if (dns->time.tv_nsec >= |
| | | 1000000000) { // nanoseconds have overflowed into seconds |
| | | dns->time.tv_sec += 1; |
| | | dns->time.tv_nsec -= 1000000000; |
| | | } |
| | | } |
| | | if (run.tv_sec == |
| | | -1) { // if test has failed 3 times, set time taken |
| | | // to 3s as penalty |
| | | dns->errors++; |
| | | } else { |
| | | dns->time.tv_sec += run.tv_sec; |
| | | dns->time.tv_nsec += run.tv_nsec; |
| | | if (dns->time.tv_nsec >= |
| | | 1000000000) { // nanoseconds have overflowed into seconds |
| | | dns->time.tv_sec += 1; |
| | | dns->time.tv_nsec -= 1000000000; |
| | | } |
| | | } |
| | | tests_done++; |
| | | curr = curr->next; |
| | | } |