/* * This file is part of the DiscImageChef Remote Server. * Copyright (c) 2019 Natalia Portillo. * * 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, version 3. * * 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, see . */ #include "dicmote.h" #include #include #include #include #include #include #include // TODO: Packet for NOP int main() { struct ifaddrs* ifa; struct ifaddrs* ifa_start; int ret; char ipv4Address[INET_ADDRSTRLEN]; int sockfd, cli_sock; struct sockaddr_in serv_addr, cli_addr; socklen_t cli_len; struct utsname utsname; DicPacketHello * pkt_server_hello, *pkt_client_hello; DicPacketHeader* pkt_hdr; ssize_t recv_size; char* dummy_buf; int skip_next_hdr; struct DeviceInfoList* deviceInfoList; printf("DiscImageChef Remote Server %s\n", DICMOTE_VERSION); printf("Copyright (C) 2019 Natalia Portillo\n"); ret = uname(&utsname); if(ret) { printf("Error %d getting system version.\n", errno); return 1; } printf("Running under %s %s (%s).\n", utsname.sysname, utsname.release, utsname.machine); printf("Opening socket.\n"); sockfd = socket(AF_INET, SOCK_STREAM, 0); if(sockfd < 0) { printf("Error %d opening socket.\n", errno); return 1; } serv_addr.sin_family = AF_INET; serv_addr.sin_addr.s_addr = INADDR_ANY; serv_addr.sin_port = htons(DICMOTE_PORT); if(bind(sockfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) < 0) { printf("Error %d binding socket.\n", errno); close(sockfd); return 1; } ret = getifaddrs(&ifa); if(ret) { printf("Error %d enumerating interfaces\n", errno); return 1; } ifa_start = ifa; printf("Available addresses:\n"); while(ifa != NULL) { if(ifa->ifa_addr && ifa->ifa_addr->sa_family == AF_INET) { inet_ntop(AF_INET, &((struct sockaddr_in*)ifa->ifa_addr)->sin_addr, ipv4Address, INET_ADDRSTRLEN); printf("%s port %d\n", ipv4Address, DICMOTE_PORT); } ifa = ifa->ifa_next; } freeifaddrs(ifa_start); ret = listen(sockfd, 1); if(ret) { printf("Error %d listening.\n", errno); close(sockfd); return 1; } pkt_server_hello = malloc(sizeof(DicPacketHello)); if(!pkt_server_hello) { printf("Fatal error %d allocating memory.\n", errno); close(sockfd); return 1; } memset(pkt_server_hello, 0, sizeof(DicPacketHello)); pkt_server_hello->hdr.id = DICMOTE_PACKET_ID; pkt_server_hello->hdr.len = sizeof(DicPacketHello); pkt_server_hello->hdr.version = DICMOTE_PACKET_VERSION; pkt_server_hello->hdr.packet_type = DICMOTE_PACKET_TYPE_HELLO; strncpy(pkt_server_hello->application, DICMOTE_NAME, sizeof(DICMOTE_NAME)); strncpy(pkt_server_hello->version, DICMOTE_VERSION, sizeof(DICMOTE_VERSION)); pkt_server_hello->max_protocol = DICMOTE_PROTOCOL_MAX; strncpy(pkt_server_hello->sysname, utsname.sysname, 255); strncpy(pkt_server_hello->release, utsname.release, 255); strncpy(pkt_server_hello->machine, utsname.machine, 255); for(;;) { printf("\n"); printf("Waiting for a client...\n"); cli_len = sizeof(cli_addr); cli_sock = accept(sockfd, (struct sockaddr*)&cli_addr, &cli_len); if(cli_sock < 0) { printf("Error %d accepting incoming connection.\n", errno); close(sockfd); return 1; } inet_ntop(AF_INET, &cli_addr.sin_addr, ipv4Address, INET_ADDRSTRLEN); printf("Client %s connected successfully.\n", ipv4Address); write(cli_sock, pkt_server_hello, sizeof(DicPacketHello)); pkt_hdr = malloc(sizeof(DicPacketHeader)); if(!pkt_hdr) { printf("Fatal error %d allocating memory.\n", errno); close(cli_sock); close(sockfd); free(pkt_server_hello); return 1; } recv_size = recv(cli_sock, pkt_hdr, sizeof(DicPacketHeader), MSG_PEEK); if(recv_size < 0) { printf("Error %d reading response from client.\n", errno); free(pkt_hdr); close(cli_sock); continue; } if(recv_size == 0) { printf("Client closed connection.\n"); free(pkt_hdr); close(cli_sock); continue; } if(pkt_hdr->id != DICMOTE_PACKET_ID) { printf("Received data is not a correct dicremote packet, closing connection...\n"); free(pkt_hdr); close(cli_sock); continue; } if(pkt_hdr->version != DICMOTE_PACKET_VERSION) { printf("Unrecognized packet version, closing connection...\n"); free(pkt_hdr); close(cli_sock); continue; } if(pkt_hdr->packet_type != DICMOTE_PACKET_TYPE_HELLO) { printf("Expecting hello packet type, received type %d, closing connection...\n", pkt_hdr->packet_type); free(pkt_hdr); close(cli_sock); continue; } pkt_client_hello = malloc(pkt_hdr->len); if(!pkt_client_hello) { printf("Fatal error %d allocating memory for packet, closing connection...\n", errno); free(pkt_hdr); close(cli_sock); continue; } recv_size = recv(cli_sock, pkt_client_hello, pkt_hdr->len, 0); if(recv_size != pkt_hdr->len) { printf("Expected %d bytes of packet, got %ld, closing connection...\n", pkt_hdr->len, recv_size); close(cli_sock); free(pkt_hdr); free(pkt_client_hello); continue; } printf("Client application: %s %s\n", pkt_client_hello->application, pkt_client_hello->version); printf("Client operating system: %s %s (%s)\n", pkt_client_hello->sysname, pkt_client_hello->release, pkt_client_hello->machine); printf("Client maximum protocol: %d\n", pkt_client_hello->max_protocol); free(pkt_client_hello); skip_next_hdr = 0; for(;;) { if(skip_next_hdr) { dummy_buf = malloc(pkt_hdr->len); if(!dummy_buf) { printf("Fatal error %d allocating memory for packet, closing connection...\n", errno); free(pkt_hdr); close(cli_sock); continue; } recv(cli_sock, dummy_buf, pkt_hdr->len, 0); free(dummy_buf); skip_next_hdr = 0; } recv_size = recv(cli_sock, pkt_hdr, sizeof(DicPacketHeader), MSG_PEEK); if(recv_size < 0) { printf("Error %d reading response from client, closing connection...\n", errno); close(cli_sock); free(pkt_hdr); break; } if(recv_size == 0) { printf("Client closed connection, closing connection...\n"); close(cli_sock); free(pkt_hdr); break; } if(pkt_hdr->id != DICMOTE_PACKET_ID) { printf("Received data is not a correct dicremote packet, closing connection...\n"); close(cli_sock); free(pkt_hdr); break; } if(pkt_hdr->version != DICMOTE_PACKET_VERSION) { printf("Unrecognized packet version, skipping...\n"); skip_next_hdr = 1; continue; } switch(pkt_hdr->packet_type) { case DICMOTE_PACKET_TYPE_HELLO: printf("Received hello packet out of order, skipping...\n"); skip_next_hdr = 1; continue; case DICMOTE_PACKET_TYPE_COMMAND_LIST_DEVICES: deviceInfoList = ListDevices(); // Packet only contains header so, dummy dummy_buf = malloc(pkt_hdr->len); if(!dummy_buf) { printf("Fatal error %d allocating memory for packet, closing connection...\n", errno); free(pkt_hdr); close(cli_sock); continue; } recv(cli_sock, dummy_buf, pkt_hdr->len, 0); free(dummy_buf); if(!deviceInfoList) { printf("Could not get device list, continuing...\n"); continue; } printf("List devices not yet implemented, skipping...\n"); continue; case DICMOTE_PACKET_TYPE_RESPONSE_LIST_DEVICES: printf("Received response packet?! You should certainly not do that...\n"); skip_next_hdr = 1; continue; default: printf("Received unrecognized packet with type %d, skipping...\n", pkt_hdr->packet_type); skip_next_hdr = 1; continue; } } } }