/** * FreeRDP: A Remote Desktop Protocol Client * Transmission Control Protocol (TCP) * * Copyright 2011 Vic Lee * Copyright 2011 Marc-Andre Moreau * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "tcp.h" void tcp_get_ip_address(rdpTcp * tcp) { uint8* ip; socklen_t length; struct sockaddr_in sockaddr; length = sizeof(sockaddr); if (getsockname(tcp->sockfd, (struct sockaddr*) &sockaddr, &length) == 0) { ip = (uint8*) (&sockaddr.sin_addr); snprintf(tcp->ip_address, sizeof(tcp->ip_address), "%d.%d.%d.%d", ip[0], ip[1], ip[2], ip[3]); } else strncpy(tcp->ip_address, "127.0.0.1", sizeof(tcp->ip_address)); tcp->ip_address[sizeof(tcp->ip_address) - 1] = 0; tcp->settings->ipv6 = 0; tcp->settings->ip_address = tcp->ip_address; } void tcp_get_mac_address(rdpTcp * tcp) { #ifdef LINUX uint8* mac; struct ifreq if_req; struct if_nameindex* ni; ni = if_nameindex(); mac = tcp->mac_address; while (ni->if_name != NULL) { if (strcmp(ni->if_name, "lo") != 0) break; ni++; } strncpy(if_req.ifr_name, ni->if_name, IF_NAMESIZE); if (ioctl(tcp->sockfd, SIOCGIFHWADDR, &if_req) != 0) { printf("failed to obtain MAC address\n"); return; } memmove((void*) mac, (void*) &if_req.ifr_ifru.ifru_hwaddr.sa_data[0], 6); #endif /* printf("MAC: %02X:%02X:%02X:%02X:%02X:%02X\n", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); */ } boolean tcp_connect(rdpTcp* tcp, const uint8* hostname, uint16 port) { int status; int sockfd = -1; char servname[10]; struct addrinfo hints = { 0 }; struct addrinfo * res, * ai; hints.ai_family = AF_UNSPEC; hints.ai_socktype = SOCK_STREAM; snprintf(servname, sizeof(servname), "%d", port); status = getaddrinfo(hostname, servname, &hints, &res); if (status != 0) { printf("transport_connect: getaddrinfo (%s)\n", gai_strerror(status)); return False; } for (ai = res; ai; ai = ai->ai_next) { sockfd = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol); if (sockfd < 0) continue; if (connect(sockfd, ai->ai_addr, ai->ai_addrlen) == 0) { printf("connected to %s:%s\n", hostname, servname); break; } sockfd = -1; } freeaddrinfo(res); if (sockfd == -1) { printf("unable to connect to %s:%s\n", hostname, servname); return False; } tcp->sockfd = sockfd; tcp_get_ip_address(tcp); tcp_get_mac_address(tcp); return True; } int tcp_read(rdpTcp* tcp, uint8* data, int length) { int status; status = recv(tcp->sockfd, data, length, 0); if (status < 0) { if (errno == EAGAIN || errno == EWOULDBLOCK) return 0; perror("recv"); return -1; } return status; } int tcp_write(rdpTcp* tcp, uint8* data, int length) { int status; int sent = 0; while (sent < length) { status = send(tcp->sockfd, data, (length - sent), MSG_NOSIGNAL); if (status < 0) { if (errno == EAGAIN || errno == EWOULDBLOCK) continue; perror("send"); return -1; } else { sent += status; data += status; } } return sent; } boolean tcp_disconnect(rdpTcp * tcp) { if (tcp->sockfd != -1) { close(tcp->sockfd); tcp->sockfd = -1; } return True; } boolean tcp_set_blocking_mode(rdpTcp* tcp, boolean blocking) { int flags; flags = fcntl(tcp->sockfd, F_GETFL); if (flags == -1) { printf("transport_configure_sockfd: fcntl failed.\n"); return False; } if (blocking == True) { /* blocking */ fcntl(tcp->sockfd, F_SETFL, flags & ~(O_NONBLOCK)); } else { /* non-blocking */ fcntl(tcp->sockfd, F_SETFL, flags | O_NONBLOCK); } return True; } rdpTcp* tcp_new(rdpSettings* settings) { rdpTcp* tcp = (rdpTcp*) xzalloc(sizeof(rdpTcp)); if (tcp != NULL) { tcp->sockfd = -1; tcp->settings = settings; tcp->connect = tcp_connect; tcp->disconnect = tcp_disconnect; tcp->set_blocking_mode = tcp_set_blocking_mode; } return tcp; } void tcp_free(rdpTcp* tcp) { if (tcp != NULL) { xfree(tcp); } }