// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "tools/android/forwarder2/command.h" #include #include #include #include #include "base/logging.h" #include "base/posix/safe_strerror.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_piece.h" #include "tools/android/forwarder2/socket.h" using base::StringPiece; namespace { // Command format: // : // // Where: // is a 5-chars zero-padded ASCII decimal integer // matching the target port for the command (e.g. // '08080' for port 8080) // is a 3-char zero-padded ASCII decimal integer // matching a command::Type value (e.g. 002 for // ACK). // The column (:) is used as a separator for easier reading. const int kPortStringSize = 5; const int kCommandTypeStringSize = 2; // Command string size also includes the ':' separator char. const int kCommandStringSize = kPortStringSize + kCommandTypeStringSize + 1; } // namespace namespace forwarder2 { bool ReadCommand(Socket* socket, int* port_out, command::Type* command_type_out) { char command_buffer[kCommandStringSize + 1]; // To make logging easier. command_buffer[kCommandStringSize] = '\0'; int bytes_read = socket->ReadNumBytes(command_buffer, kCommandStringSize); if (bytes_read != kCommandStringSize) { if (bytes_read < 0) LOG(ERROR) << "Read() error: " << base::safe_strerror(errno); else if (!bytes_read) LOG(ERROR) << "Read() error, endpoint was unexpectedly closed."; else LOG(ERROR) << "Read() error, not enough data received from the socket."; return false; } StringPiece port_str(command_buffer, kPortStringSize); if (!StringToInt(port_str, port_out)) { LOG(ERROR) << "Could not parse the command port string: " << port_str; return false; } StringPiece command_type_str( &command_buffer[kPortStringSize + 1], kCommandTypeStringSize); int command_type; if (!StringToInt(command_type_str, &command_type)) { LOG(ERROR) << "Could not parse the command type string: " << command_type_str; return false; } *command_type_out = static_cast(command_type); return true; } bool SendCommand(command::Type command, int port, Socket* socket) { char buffer[kCommandStringSize + 1]; int len = snprintf(buffer, sizeof(buffer), "%05d:%02d", port, command); CHECK_EQ(len, kCommandStringSize); // Write the full command minus the leading \0 char. return socket->WriteNumBytes(buffer, len) == len; } bool ReceivedCommand(command::Type command, Socket* socket) { int port; command::Type received_command; if (!ReadCommand(socket, &port, &received_command)) return false; return received_command == command; } } // namespace forwarder