/* * 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; DicPacketResListDevs* deviceInfoResponsePacket; int i; uint64_t n; DicPacketNop* pkt_nop; DicPacketCmdOpen* pkt_dev_open; int device_fd; 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); pkt_nop = malloc(sizeof(DicPacketNop)); if(!pkt_nop) { printf("Fatal error %d allocating memory.\n", errno); close(sockfd); return 1; } memset(pkt_nop, 0, sizeof(DicPacketNop)); pkt_nop->hdr.id = DICMOTE_PACKET_ID; pkt_nop->hdr.len = sizeof(DicPacketNop); pkt_nop->hdr.version = DICMOTE_PACKET_VERSION; pkt_nop->hdr.packet_type = DICMOTE_PACKET_TYPE_NOP; 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: pkt_nop->reason_code = DICMOTE_PACKET_NOP_REASON_OOO; memset(&pkt_nop->reason, 0, 256); strncpy(pkt_nop->reason, "Received hello packet out of order, skipping...", 256); write(cli_sock, pkt_nop, sizeof(DicPacketNop)); printf("%s...\n", pkt_nop->reason); 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) { pkt_nop->reason_code = DICMOTE_PACKET_NOP_REASON_ERROR_LIST_DEVICES; memset(&pkt_nop->reason, 0, 256); strncpy(pkt_nop->reason, "Could not get device list, continuing...", 256); write(cli_sock, pkt_nop, sizeof(DicPacketNop)); printf("%s...\n", pkt_nop->reason); continue; } deviceInfoResponsePacket = malloc(sizeof(DicPacketResListDevs)); deviceInfoResponsePacket->devices = DeviceInfoListCount(deviceInfoList); n = sizeof(DicPacketResListDevs) + deviceInfoResponsePacket->devices * sizeof(DeviceInfo); dummy_buf = malloc(n); ((DicPacketResListDevs*)dummy_buf)->hdr.len = n; ((DicPacketResListDevs*)dummy_buf)->devices = deviceInfoResponsePacket->devices; free(deviceInfoResponsePacket); deviceInfoResponsePacket = (DicPacketResListDevs*)dummy_buf; dummy_buf = NULL; deviceInfoResponsePacket->hdr.id = DICMOTE_PACKET_ID; deviceInfoResponsePacket->hdr.version = DICMOTE_PACKET_VERSION; deviceInfoResponsePacket->hdr.packet_type = DICMOTE_PACKET_TYPE_RESPONSE_LIST_DEVICES; // Save list start dummy_buf = (char*)deviceInfoList; long off = sizeof(DicPacketResListDevs); while(deviceInfoList) { memcpy(((char*)deviceInfoResponsePacket) + off, &deviceInfoList->this, sizeof(DeviceInfo)); deviceInfoList = deviceInfoList->next; off += sizeof(DeviceInfo); } deviceInfoList = (struct DeviceInfoList*)dummy_buf; FreeDeviceInfoList(deviceInfoList); write(cli_sock, deviceInfoResponsePacket, deviceInfoResponsePacket->hdr.len); free(deviceInfoResponsePacket); printf("List devices not yet implemented, skipping...\n"); continue; case DICMOTE_PACKET_TYPE_RESPONSE_LIST_DEVICES: pkt_nop->reason_code = DICMOTE_PACKET_NOP_REASON_OOO; memset(&pkt_nop->reason, 0, 256); strncpy(pkt_nop->reason, "Received response packet?! You should certainly not do that...", 256); write(cli_sock, pkt_nop, sizeof(DicPacketNop)); printf("%s...\n", pkt_nop->reason); skip_next_hdr = 1; continue; case DICMOTE_PACKET_TYPE_COMMAND_OPEN_DEVICE: pkt_dev_open = malloc(pkt_hdr->len); if(!pkt_dev_open) { printf("Fatal error %d allocating memory for packet, closing connection...\n", errno); free(pkt_hdr); close(cli_sock); continue; } recv(cli_sock, pkt_dev_open, pkt_hdr->len, 0); device_fd = DeviceOpen(pkt_dev_open->device_path); pkt_nop->reason_code = device_fd == -1 ? DICMOTE_PACKET_NOP_REASON_OPEN_ERROR : DICMOTE_PACKET_NOP_REASON_OPEN_OK; pkt_nop->errorNo = errno; memset(&pkt_nop->reason, 0, 256); write(cli_sock, pkt_nop, sizeof(DicPacketNop)); free(pkt_dev_open); continue; default: pkt_nop->reason_code = DICMOTE_PACKET_NOP_REASON_NOT_RECOGNIZED; memset(&pkt_nop->reason, 0, 256); snprintf(pkt_nop->reason, 256, "Received unrecognized packet with type %d, skipping...", pkt_hdr->packet_type); strncpy(pkt_nop->reason, "Received unrecognized packet with type %d, skipping...", 256); write(cli_sock, pkt_nop, sizeof(DicPacketNop)); printf("%s...\n", pkt_nop->reason); skip_next_hdr = 1; continue; } } } }