diff options
author | Håkon Nessjøen <haakon.nessjoen@gmail.com> | 2011-02-21 15:53:35 +0100 |
---|---|---|
committer | Håkon Nessjøen <haakon.nessjoen@gmail.com> | 2011-02-21 15:53:35 +0100 |
commit | 0058ac3010d79bf086ddfa0be9a44ee419bf7058 (patch) | |
tree | 7044300a72c731ffc6f9d6bca94d2ef2bdd76b0b /macping.c | |
parent | d3f4e55538788ec7569c5eb4a21622c516aa0d92 (diff) | |
download | MAC-Telnet-0058ac3010d79bf086ddfa0be9a44ee419bf7058.tar.gz MAC-Telnet-0058ac3010d79bf086ddfa0be9a44ee419bf7058.zip |
Added MT_PTYPE_PING/PONG support in mactelnetd and added a macping tool. For this udp-checksum code was also added for raw packets.
Diffstat (limited to 'macping.c')
-rw-r--r-- | macping.c | 327 |
1 files changed, 327 insertions, 0 deletions
diff --git a/macping.c b/macping.c new file mode 100644 index 0000000..34ae0c6 --- /dev/null +++ b/macping.c @@ -0,0 +1,327 @@ +/* + Mac-Telnet - Connect to RouterOS routers via MAC address + Copyright (C) 2010, Håkon Nessjøen <haakon.nessjoen@gmail.com> + + This program 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 2 of the License, or + (at your option) any later version. + + This program 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, write to the Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +*/ +#include <stdlib.h> +#include <signal.h> +#include <stdio.h> +#include <arpa/inet.h> +#include <netinet/ether.h> +#include <string.h> +#include <time.h> +#include <sys/time.h> +#include <unistd.h> +#include <stdio.h> +#include <float.h> +#include "protocol.h" +#include "udp.h" +#include "devices.h" +#include "config.h" + +#define MAX_DEVICES 128 +#define MT_INTERFACE_LEN 128 + +#define PROGRAM_NAME "MAC-Ping" +#define PROGRAM_VERSION "0.1" + +static int sockfd, insockfd; + +struct mt_device { + unsigned char mac[ETH_ALEN]; + char name[MT_INTERFACE_LEN]; + int device_index; +}; + +static unsigned short ping_size = 38; +static struct mt_device devices[MAX_DEVICES]; +static int devices_count = 0; +static struct in_addr sourceip; +static struct in_addr destip; +static unsigned char dstmac[6]; + +static int ping_sent = 0; +static int pong_received = 0; +static float min_ms = FLT_MAX; +static float avg_ms = 0; +static float max_ms = 0; + +/* Protocol data direction, not used here, but obligatory for protocol.c */ +unsigned char mt_direction_fromserver = 0; + +static void print_version() { + fprintf(stderr, PROGRAM_NAME " " PROGRAM_VERSION "\n"); +} + +static void setup_devices() { + char devicename[MT_INTERFACE_LEN]; + unsigned char mac[ETH_ALEN]; + unsigned char emptymac[ETH_ALEN]; + int success; + + memset(emptymac, 0, ETH_ALEN); + + while ((success = get_macs(insockfd, devicename, MT_INTERFACE_LEN, mac))) { + if (memcmp(mac, emptymac, ETH_ALEN) != 0) { + struct mt_device *device = &(devices[devices_count]); + + memcpy(device->mac, mac, ETH_ALEN); + strncpy(device->name, devicename, MT_INTERFACE_LEN - 1); + device->name[MT_INTERFACE_LEN - 1] = '\0'; + + device->device_index = get_device_index(insockfd, devicename); + + devices_count++; + } + } +} + +long long int toddiff(struct timeval *tod1, struct timeval *tod2) +{ + long long t1, t2; + t1 = tod1->tv_sec * 1000000 + tod1->tv_usec; + t2 = tod2->tv_sec * 1000000 + tod2->tv_usec; + return t1 - t2; +} + +void display_results() { + int percent = (int)((100.f/ping_sent) * pong_received); + if (percent > 100) + percent = 0; + + if (percent < 0) + percent = 0; + + if (min_ms == FLT_MAX) + min_ms = 0; + + fprintf(stderr, "\n"); + fprintf(stderr, "%d packets transmitted, %d packets received, %d%% packet loss\n", ping_sent, pong_received, 100 - percent); + fprintf(stderr, "round-trip min/avg/max = %.2f/%.2f/%.2f ms\n", min_ms, avg_ms/pong_received, max_ms); + + /* For bash scripting */ + if (pong_received == 0) { + exit(1); + } + + exit(0); +} + +int main(int argc, char **argv) { + int optval = 1; + int print_help = 0; + int send_packets = 5; + int c; + struct sockaddr_in si_me; + struct mt_packet packet; + int i; + + while (1) { + c = getopt(argc, argv, "s:c:hv?"); + + if (c == -1) + break; + + switch (c) { + case 's': + ping_size = atoi(optarg) - 18; + break; + + case 'v': + print_version(); + exit(0); + break; + + case 'c': + send_packets = atoi(optarg); + break; + + case 'h': + case '?': + print_help = 1; + break; + + } + } + + if (argc - optind < 1 || print_help) { + print_version(); + fprintf(stderr, "Usage: %s <MAC> [-h] [-c <count>] [-s <packet size>]\n", argv[0]); + + if (print_help) { + fprintf(stderr, "\nParameters:\n"); + fprintf(stderr, " MAC MAC-Address of the RouterOS/mactelnetd device.\n"); + fprintf(stderr, " -s Specify size of ping packet.\n"); + fprintf(stderr, " -c Number of packets to send. (0 = unlimited)\n"); + fprintf(stderr, " -h This help.\n"); + fprintf(stderr, "\n"); + } + return 1; + } + + if (ping_size > ETH_FRAME_LEN - 42) { + fprintf(stderr, "Packet size must be between 18 and %d\n", ETH_FRAME_LEN - 42 + 18); + exit(1); + } + + if (geteuid() != 0) { + fprintf(stderr, "You need to have root privileges to use %s.\n", argv[0]); + return 1; + } + + ether_aton_r(argv[optind], (struct ether_addr *)dstmac); + + /* Open a UDP socket handle */ + sockfd = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ALL)); + if (sockfd < 0) { + perror("sockfd"); + return 1; + } + + insockfd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); + if (insockfd < 0) { + perror("insockfd"); + return 1; + } + + /* Set initialize address/port */ + memset((char *) &si_me, 0, sizeof(si_me)); + si_me.sin_family = AF_INET; + si_me.sin_port = htons(MT_MACTELNET_PORT); + si_me.sin_addr.s_addr = htonl(INADDR_ANY); + + setsockopt(insockfd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof (optval)); + + /* Bind to specified address/port */ + if (bind(insockfd, (struct sockaddr *)&si_me, sizeof(si_me))==-1) { + fprintf(stderr, "Error binding to %s:%d\n", inet_ntoa(si_me.sin_addr), MT_MNDP_PORT); + return 1; + } + + /* Listen address*/ + inet_pton(AF_INET, (char *)"0.0.0.0", &sourceip); + + /* Set up global info about the connection */ + inet_pton(AF_INET, (char *)"255.255.255.255", &destip); + + srand(time(NULL)); + + setup_devices(); + + if (ping_size < sizeof(struct timeval)) { + ping_size = sizeof(struct timeval); + } + + signal(SIGINT, display_results); + + for (i = 0; i < send_packets || send_packets == 0; ++i) { + fd_set read_fds; + static struct timeval lasttimestamp; + int reads, result; + struct timeval timeout; + int ii; + int sent = 0; + int waitforpacket; + struct timeval timestamp; + unsigned char pingdata[1500]; + + gettimeofday(×tamp, NULL); + memcpy(pingdata, ×tamp, sizeof(timestamp)); + for (ii = sizeof(timestamp); ii < ping_size; ++ii) { + pingdata[ii] = rand() % 256; + } + + for (ii = 0; ii < devices_count; ++ii) { + struct mt_device *device = &devices[ii]; + + init_pingpacket(&packet, device->mac, dstmac); + add_packetdata(&packet, pingdata, ping_size); + result = send_custom_udp(sockfd, device->device_index, device->mac, dstmac, &sourceip, MT_MACTELNET_PORT, &destip, MT_MACTELNET_PORT, packet.data, packet.size); + + if (result > 0) { + sent++; + } + + } + if (sent == 0) { + fprintf(stderr, "Error sending packet.\n"); + continue; + } + ping_sent++; + + FD_ZERO(&read_fds); + FD_SET(insockfd, &read_fds); + + timeout.tv_sec = 1; + timeout.tv_usec = 0; + + waitforpacket = 1; + + while (waitforpacket) { + /* Wait for data or timeout */ + reads = select(insockfd+1, &read_fds, NULL, NULL, &timeout); + if (reads > 0) { + unsigned char buff[1500]; + struct sockaddr_in saddress; + unsigned int slen = sizeof(saddress); + struct mt_mactelnet_hdr pkthdr; + + result = recvfrom(insockfd, buff, 1500, 0, (struct sockaddr *)&saddress, &slen); + parse_packet(buff, &pkthdr); + + /* TODO: Check that we are the receiving host */ + if (pkthdr.ptype == MT_PTYPE_PONG) { + struct timeval pongtimestamp; + struct timeval nowtimestamp; + + waitforpacket = 0; + gettimeofday(&nowtimestamp, NULL); + + memcpy(&pongtimestamp, pkthdr.data - 4, sizeof(pongtimestamp)); + if (memcmp(pkthdr.data - 4, pingdata, ping_size) == 0) { + float diff = toddiff(&nowtimestamp, &pongtimestamp) / 1000.0f; + + if (diff < min_ms) + min_ms = diff; + + if (diff > max_ms) + max_ms = diff; + + avg_ms += diff; + + printf("%s %d byte, ping time %.2f ms%s\n", ether_ntoa((struct ether_addr *)&(pkthdr.srcaddr)), result, diff, (char *)(memcmp(&pongtimestamp,&lasttimestamp,sizeof(lasttimestamp)) == 0 ? " DUP" : "")); + } else { + printf("%s Reply of %d bytes of unequal data\n", ether_ntoa((struct ether_addr *)&(pkthdr.srcaddr)), result); + } + pong_received++; + memcpy(&lasttimestamp, &pongtimestamp, sizeof(pongtimestamp)); + sleep(1); + } else { + /* Wait for the correct packet */ + continue; + } + } else { + waitforpacket = 0; + fprintf(stderr, "%s ping timeout\n", ether_ntoa((struct ether_addr *)&dstmac)); + } + } + } + + /* Display statistics and exit */ + display_results(); + + return 0; +} |