Files
clang-p2996/lldb/tools/debugserver/source/RNBSocket.cpp
Greg Clayton 95bf0fd3ab Added the ability to get a broadcaster event name for a given broadcaster
event.

Modified the ProcessInfo structure to contain all process arguments. Using the
new function calls on MacOSX allows us to see the full process name, not just
the first 16 characters. 

Added a new platform command: "platform process info <pid> [<pid> <pid> ...]"
that can be used to get detailed information for a process including all 
arguments, user and group info and more.

llvm-svn: 128694
2011-04-01 00:29:43 +00:00

303 lines
8.3 KiB
C++

//===-- RNBSocket.cpp -------------------------------------------*- C++ -*-===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// Created by Greg Clayton on 12/12/07.
//
//===----------------------------------------------------------------------===//
#include "RNBSocket.h"
#include <arpa/inet.h>
#include <errno.h>
#include <fcntl.h>
#include <netdb.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <termios.h>
#include "DNBLog.h"
#include "DNBError.h"
#if defined (__arm__)
#include "lockdown.h"
#endif
/* Once we have a RNBSocket object with a port # specified,
this function is called to wait for an incoming connection.
This function blocks while waiting for that connection. */
rnb_err_t
RNBSocket::Listen (in_port_t listen_port_num)
{
//DNBLogThreadedIf(LOG_RNB_COMM, "%8u RNBSocket::%s called", (uint32_t)m_timer.ElapsedMicroSeconds(true), __FUNCTION__);
// Disconnect without saving errno
Disconnect (false);
DNBError err;
int listen_port = ::socket (AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (listen_port == -1)
err.SetError(errno, DNBError::POSIX);
if (err.Fail() || DNBLogCheckLogBit(LOG_RNB_COMM))
err.LogThreaded("::socket ( domain = AF_INET, type = SOCK_STREAM, protocol = IPPROTO_TCP ) => socket = %i", listen_port);
if (err.Fail())
return rnb_err;
// enable local address reuse
SetSocketOption (listen_port, SOL_SOCKET, SO_REUSEADDR, 1);
struct sockaddr_in sa;
::memset (&sa, 0, sizeof sa);
sa.sin_len = sizeof sa;
sa.sin_family = AF_INET;
sa.sin_port = htons (listen_port_num);
sa.sin_addr.s_addr = htonl (INADDR_ANY);
int error = ::bind (listen_port, (struct sockaddr *) &sa, sizeof(sa));
if (error == -1)
err.SetError(errno, DNBError::POSIX);
if (err.Fail() || DNBLogCheckLogBit(LOG_RNB_COMM))
err.LogThreaded("::bind ( socket = %i, (struct sockaddr *) &sa, sizeof(sa)) )", listen_port);
if (err.Fail())
{
ClosePort (listen_port, false);
return rnb_err;
}
error = ::listen (listen_port, 1);
if (error == -1)
err.SetError(errno, DNBError::POSIX);
if (err.Fail() || DNBLogCheckLogBit(LOG_RNB_COMM))
err.LogThreaded("::listen ( socket = %i, backlog = 1 )", listen_port);
if (err.Fail())
{
ClosePort (listen_port, false);
return rnb_err;
}
m_conn_port = ::accept (listen_port, NULL, 0);
if (m_conn_port == -1)
err.SetError(errno, DNBError::POSIX);
if (err.Fail() || DNBLogCheckLogBit(LOG_RNB_COMM))
err.LogThreaded("::accept ( socket = %i, address = NULL, address_len = 0 )", listen_port);
if (err.Fail())
{
ClosePort (listen_port, false);
return rnb_err;
}
else
{
// We are done with the listen port
ClosePort (listen_port, false);
// Keep our TCP packets coming without any delays.
SetSocketOption (m_conn_port, IPPROTO_TCP, TCP_NODELAY, 1);
}
return rnb_success;
}
rnb_err_t
RNBSocket::Connect (const char *host, uint16_t port)
{
Disconnect (false);
// Create the socket
m_conn_port = ::socket (AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (m_conn_port == -1)
return rnb_err;
// Enable local address reuse
SetSocketOption (m_conn_port, SOL_SOCKET, SO_REUSEADDR, 1);
struct sockaddr_in sa;
::memset (&sa, 0, sizeof (sa));
sa.sin_family = AF_INET;
sa.sin_port = htons (port);
if (host == NULL)
host = "localhost";
int inet_pton_result = ::inet_pton (AF_INET, host, &sa.sin_addr);
if (inet_pton_result <= 0)
{
struct hostent *host_entry = gethostbyname (host);
if (host_entry)
{
std::string host_str (::inet_ntoa (*(struct in_addr *)*host_entry->h_addr_list));
inet_pton_result = ::inet_pton (AF_INET, host_str.c_str(), &sa.sin_addr);
if (inet_pton_result <= 0)
{
Disconnect (false);
return rnb_err;
}
}
}
if (-1 == ::connect (m_conn_port, (const struct sockaddr *)&sa, sizeof(sa)))
{
Disconnect (false);
return rnb_err;
}
// Keep our TCP packets coming without any delays.
SetSocketOption (m_conn_port, IPPROTO_TCP, TCP_NODELAY, 1);
return rnb_success;
}
#if defined (__arm__)
rnb_err_t
RNBSocket::ConnectToService()
{
DNBLog("Connecting to com.apple.%s service...", DEBUGSERVER_PROGRAM_NAME);
// Disconnect from any previous connections
Disconnect(false);
m_conn_port = ::lockdown_checkin (NULL, NULL);
if (m_conn_port == -1)
{
DNBLogThreadedIf(LOG_RNB_COMM, "::lockdown_checkin(NULL, NULL) failed");
return rnb_not_connected;
}
m_conn_port_from_lockdown = true;
return rnb_success;
}
#endif
rnb_err_t
RNBSocket::OpenFile (const char *path)
{
DNBError err;
m_conn_port = open (path, O_RDWR);
if (m_conn_port == -1)
{
err.SetError(errno, DNBError::POSIX);
err.LogThreaded ("can't open file '%s'", path);
return rnb_not_connected;
}
else
{
struct termios stdin_termios;
if (::tcgetattr (m_conn_port, &stdin_termios) == 0)
{
stdin_termios.c_lflag &= ~ECHO; // Turn off echoing
stdin_termios.c_lflag &= ~ICANON; // Get one char at a time
::tcsetattr (m_conn_port, TCSANOW, &stdin_termios);
}
}
return rnb_success;
}
int
RNBSocket::SetSocketOption(int fd, int level, int option_name, int option_value)
{
return ::setsockopt(fd, level, option_name, &option_value, sizeof(option_value));
}
rnb_err_t
RNBSocket::Disconnect (bool save_errno)
{
if (m_conn_port_from_lockdown)
m_conn_port_from_lockdown = false;
return ClosePort (m_conn_port, save_errno);
}
rnb_err_t
RNBSocket::Read (std::string &p)
{
char buf[1024];
p.clear();
// Note that BUF is on the stack so we must be careful to keep any
// writes to BUF from overflowing or we'll have security issues.
if (m_conn_port == -1)
return rnb_err;
//DNBLogThreadedIf(LOG_RNB_COMM, "%8u RNBSocket::%s calling read()", (uint32_t)m_timer.ElapsedMicroSeconds(true), __FUNCTION__);
DNBError err;
int bytesread = read (m_conn_port, buf, sizeof (buf));
if (bytesread <= 0)
err.SetError(errno, DNBError::POSIX);
else
p.append(buf, bytesread);
if (err.Fail() || DNBLogCheckLogBit(LOG_RNB_COMM))
err.LogThreaded("::read ( %i, %p, %zu ) => %i", m_conn_port, buf, sizeof (buf), bytesread);
// Our port went away - we have to mark this so IsConnected will return the truth.
if (bytesread == 0)
{
m_conn_port = -1;
return rnb_not_connected;
}
else if (bytesread == -1)
{
m_conn_port = -1;
return rnb_err;
}
// Strip spaces from the end of the buffer
while (!p.empty() && isspace (p[p.size() - 1]))
p.erase (p.size () - 1);
// Most data in the debugserver packets valid printable characters...
DNBLogThreadedIf(LOG_RNB_COMM, "read: %s", p.c_str());
return rnb_success;
}
rnb_err_t
RNBSocket::Write (const void *buffer, size_t length)
{
if (m_conn_port == -1)
return rnb_err;
DNBError err;
int bytessent = send (m_conn_port, buffer, length, 0);
if (bytessent < 0)
err.SetError(errno, DNBError::POSIX);
if (err.Fail() || DNBLogCheckLogBit(LOG_RNB_COMM))
err.LogThreaded("::send ( socket = %i, buffer = %p, length = %zu, flags = 0 ) => %i", m_conn_port, buffer, length, bytessent);
if (bytessent < 0)
return rnb_err;
if (bytessent != length)
return rnb_err;
DNBLogThreadedIf(LOG_RNB_PACKETS, "putpkt: %*s", length, (char *)buffer); // All data is string based in debugserver, so this is safe
DNBLogThreadedIf(LOG_RNB_COMM, "sent: %*s", length, (char *)buffer);
return rnb_success;
}
rnb_err_t
RNBSocket::ClosePort (int& fd, bool save_errno)
{
int close_err = 0;
if (fd > 0)
{
errno = 0;
close_err = close (fd);
fd = -1;
}
return close_err != 0 ? rnb_err : rnb_success;
}