2011-03-09 06:40:15 +08:00
|
|
|
//===-- Platform.cpp --------------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "lldb/Target/Platform.h"
|
|
|
|
|
|
|
|
// C Includes
|
|
|
|
// C++ Includes
|
|
|
|
// Other libraries and framework includes
|
|
|
|
// Project includes
|
|
|
|
#include "lldb/Core/Error.h"
|
|
|
|
#include "lldb/Core/PluginManager.h"
|
|
|
|
#include "lldb/Host/FileSpec.h"
|
2011-03-19 09:12:21 +08:00
|
|
|
#include "lldb/Host/Host.h"
|
2011-03-09 06:40:15 +08:00
|
|
|
#include "lldb/Target/Target.h"
|
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
// Use a singleton function for g_local_platform_sp to avoid init
|
|
|
|
// constructors since LLDB is often part of a shared library
|
|
|
|
static PlatformSP&
|
|
|
|
GetDefaultPlatformSP ()
|
|
|
|
{
|
|
|
|
static PlatformSP g_default_platform_sp;
|
|
|
|
return g_default_platform_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Mutex &
|
|
|
|
GetConnectedPlatformListMutex ()
|
|
|
|
{
|
|
|
|
static Mutex g_remote_connected_platforms_mutex (Mutex::eMutexTypeRecursive);
|
|
|
|
return g_remote_connected_platforms_mutex;
|
|
|
|
}
|
|
|
|
static std::vector<PlatformSP> &
|
|
|
|
GetConnectedPlatformList ()
|
|
|
|
{
|
|
|
|
static std::vector<PlatformSP> g_remote_connected_platforms;
|
|
|
|
return g_remote_connected_platforms;
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
/// Get the native host platform plug-in.
|
|
|
|
///
|
|
|
|
/// There should only be one of these for each host that LLDB runs
|
|
|
|
/// upon that should be statically compiled in and registered using
|
|
|
|
/// preprocessor macros or other similar build mechanisms.
|
|
|
|
///
|
|
|
|
/// This platform will be used as the default platform when launching
|
|
|
|
/// or attaching to processes unless another platform is specified.
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
PlatformSP
|
|
|
|
Platform::GetDefaultPlatform ()
|
|
|
|
{
|
|
|
|
return GetDefaultPlatformSP ();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Platform::SetDefaultPlatform (const lldb::PlatformSP &platform_sp)
|
|
|
|
{
|
|
|
|
// The native platform should use its static void Platform::Initialize()
|
|
|
|
// function to register itself as the native platform.
|
|
|
|
GetDefaultPlatformSP () = platform_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
Error
|
2011-03-23 08:09:55 +08:00
|
|
|
Platform::GetFile (const FileSpec &platform_file,
|
|
|
|
const UUID *uuid_ptr,
|
|
|
|
FileSpec &local_file)
|
2011-03-09 06:40:15 +08:00
|
|
|
{
|
|
|
|
// Default to the local case
|
|
|
|
local_file = platform_file;
|
|
|
|
return Error();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
PlatformSP
|
2011-03-19 09:12:21 +08:00
|
|
|
Platform::Create (const char *platform_name, Error &error)
|
2011-03-09 06:40:15 +08:00
|
|
|
{
|
|
|
|
PlatformCreateInstance create_callback = NULL;
|
|
|
|
lldb::PlatformSP platform_sp;
|
2011-03-19 09:12:21 +08:00
|
|
|
if (platform_name && platform_name[0])
|
2011-03-09 06:40:15 +08:00
|
|
|
{
|
|
|
|
create_callback = PluginManager::GetPlatformCreateCallbackForPluginName (platform_name);
|
|
|
|
if (create_callback)
|
|
|
|
platform_sp.reset(create_callback());
|
|
|
|
else
|
2011-03-19 09:12:21 +08:00
|
|
|
error.SetErrorStringWithFormat ("unable to find a plug-in for the platform named \"%s\"", platform_name);
|
2011-03-09 06:40:15 +08:00
|
|
|
}
|
|
|
|
else
|
2011-03-19 09:12:21 +08:00
|
|
|
error.SetErrorString ("invalid platform name");
|
2011-03-09 06:40:15 +08:00
|
|
|
return platform_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t
|
|
|
|
Platform::GetNumConnectedRemotePlatforms ()
|
|
|
|
{
|
|
|
|
Mutex::Locker locker (GetConnectedPlatformListMutex ());
|
|
|
|
return GetConnectedPlatformList().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
PlatformSP
|
|
|
|
Platform::GetConnectedRemotePlatformAtIndex (uint32_t idx)
|
|
|
|
{
|
|
|
|
PlatformSP platform_sp;
|
|
|
|
{
|
|
|
|
Mutex::Locker locker (GetConnectedPlatformListMutex ());
|
|
|
|
if (idx < GetConnectedPlatformList().size())
|
|
|
|
platform_sp = GetConnectedPlatformList ()[idx];
|
|
|
|
}
|
|
|
|
return platform_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
/// Default Constructor
|
|
|
|
//------------------------------------------------------------------
|
2011-03-19 09:12:21 +08:00
|
|
|
Platform::Platform (bool is_host) :
|
|
|
|
m_is_host (is_host),
|
|
|
|
m_is_connected (is_host), // If this is the default host platform, then we are always connected
|
|
|
|
m_os_version_set_while_connected (false),
|
|
|
|
m_system_arch_set_while_connected (false),
|
|
|
|
m_remote_url (),
|
2011-03-23 08:09:55 +08:00
|
|
|
m_remote_instance_name (),
|
2011-03-19 09:12:21 +08:00
|
|
|
m_major_os_version (UINT32_MAX),
|
|
|
|
m_minor_os_version (UINT32_MAX),
|
|
|
|
m_update_os_version (UINT32_MAX)
|
2011-03-09 06:40:15 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
/// Destructor.
|
|
|
|
///
|
|
|
|
/// The destructor is virtual since this class is designed to be
|
|
|
|
/// inherited from by the plug-in instance.
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
Platform::~Platform()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-03-19 09:12:21 +08:00
|
|
|
|
|
|
|
bool
|
|
|
|
Platform::GetOSVersion (uint32_t &major,
|
|
|
|
uint32_t &minor,
|
|
|
|
uint32_t &update)
|
|
|
|
{
|
|
|
|
bool success = m_major_os_version != UINT32_MAX;
|
|
|
|
if (IsHost())
|
|
|
|
{
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
// We have a local host platform
|
|
|
|
success = Host::GetOSVersion (m_major_os_version,
|
|
|
|
m_minor_os_version,
|
|
|
|
m_update_os_version);
|
|
|
|
m_os_version_set_while_connected = success;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We have a remote platform. We can only fetch the remote
|
|
|
|
// OS version if we are connected, and we don't want to do it
|
|
|
|
// more than once.
|
|
|
|
|
|
|
|
const bool is_connected = IsConnected();
|
|
|
|
|
|
|
|
bool fetch_os_version = false;
|
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
// We have valid OS version info, check to make sure it wasn't
|
|
|
|
// manually set prior to connecting. If it was manually set prior
|
|
|
|
// to connecting, then lets fetch the actual OS version info
|
|
|
|
// if we are now connected.
|
|
|
|
if (is_connected && !m_os_version_set_while_connected)
|
|
|
|
fetch_os_version = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We don't have valid OS version info, fetch it if we are connected
|
|
|
|
fetch_os_version = is_connected;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fetch_os_version)
|
|
|
|
{
|
|
|
|
success = FetchRemoteOSVersion ();
|
|
|
|
m_os_version_set_while_connected = success;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
major = m_major_os_version;
|
|
|
|
minor = m_minor_os_version;
|
|
|
|
update = m_update_os_version;
|
|
|
|
}
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Platform::SetOSVersion (uint32_t major,
|
|
|
|
uint32_t minor,
|
|
|
|
uint32_t update)
|
|
|
|
{
|
|
|
|
if (IsHost())
|
|
|
|
{
|
|
|
|
// We don't need anyone setting the OS version for the host platform,
|
|
|
|
// we should be able to figure it out by calling Host::GetOSVersion(...).
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We have a remote platform, allow setting the target OS version if
|
|
|
|
// we aren't connected, since if we are connected, we should be able to
|
|
|
|
// request the remote OS version from the connected platform.
|
|
|
|
if (IsConnected())
|
|
|
|
return false;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We aren't connected and we might want to set the OS version
|
|
|
|
// ahead of time before we connect so we can peruse files and
|
|
|
|
// use a local SDK or PDK cache of support files to disassemble
|
|
|
|
// or do other things.
|
|
|
|
m_major_os_version = major;
|
|
|
|
m_minor_os_version = minor;
|
|
|
|
m_update_os_version = update;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-09 06:40:15 +08:00
|
|
|
Error
|
|
|
|
Platform::ResolveExecutable (const FileSpec &exe_file,
|
|
|
|
const ArchSpec &exe_arch,
|
|
|
|
lldb::ModuleSP &exe_module_sp)
|
|
|
|
{
|
|
|
|
Error error;
|
|
|
|
if (exe_file.Exists())
|
|
|
|
{
|
|
|
|
if (exe_arch.IsValid())
|
|
|
|
{
|
|
|
|
error = ModuleList::GetSharedModule (exe_file,
|
|
|
|
exe_arch,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
exe_module_sp,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// No valid architecture was specified, ask the platform for
|
|
|
|
// the architectures that we should be using (in the correct order)
|
|
|
|
// and see if we can find a match that way
|
|
|
|
ArchSpec platform_arch;
|
|
|
|
for (uint32_t idx = 0; GetSupportedArchitectureAtIndex (idx, platform_arch); ++idx)
|
|
|
|
{
|
|
|
|
error = ModuleList::GetSharedModule (exe_file,
|
|
|
|
platform_arch,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
exe_module_sp,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
// Did we find an executable using one of the
|
|
|
|
if (error.Success() && exe_module_sp)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
error.SetErrorStringWithFormat ("'%s%s%s' does not exist",
|
|
|
|
exe_file.GetDirectory().AsCString(""),
|
|
|
|
exe_file.GetDirectory() ? "/" : "",
|
|
|
|
exe_file.GetFilename().AsCString(""));
|
|
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2011-03-19 09:12:21 +08:00
|
|
|
|
|
|
|
const ArchSpec &
|
|
|
|
Platform::GetSystemArchitecture()
|
|
|
|
{
|
|
|
|
if (IsHost())
|
|
|
|
{
|
|
|
|
if (!m_system_arch.IsValid())
|
|
|
|
{
|
|
|
|
// We have a local host platform
|
|
|
|
m_system_arch = Host::GetArchitecture();
|
|
|
|
m_system_arch_set_while_connected = m_system_arch.IsValid();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We have a remote platform. We can only fetch the remote
|
|
|
|
// system architecture if we are connected, and we don't want to do it
|
|
|
|
// more than once.
|
|
|
|
|
|
|
|
const bool is_connected = IsConnected();
|
|
|
|
|
|
|
|
bool fetch = false;
|
|
|
|
if (m_system_arch.IsValid())
|
|
|
|
{
|
|
|
|
// We have valid OS version info, check to make sure it wasn't
|
|
|
|
// manually set prior to connecting. If it was manually set prior
|
|
|
|
// to connecting, then lets fetch the actual OS version info
|
|
|
|
// if we are now connected.
|
|
|
|
if (is_connected && !m_system_arch_set_while_connected)
|
|
|
|
fetch = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// We don't have valid OS version info, fetch it if we are connected
|
|
|
|
fetch = is_connected;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fetch)
|
|
|
|
{
|
|
|
|
m_system_arch = FetchRemoteSystemArchitecture ();
|
|
|
|
m_system_arch_set_while_connected = m_system_arch.IsValid();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return m_system_arch;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-09 06:40:15 +08:00
|
|
|
Error
|
2011-03-23 08:09:55 +08:00
|
|
|
Platform::ConnectRemote (Args& args)
|
2011-03-09 06:40:15 +08:00
|
|
|
{
|
|
|
|
Error error;
|
2011-03-23 08:09:55 +08:00
|
|
|
if (IsHost())
|
|
|
|
error.SetErrorStringWithFormat ("The currently selected platform (%s) is the host platform and is always connected.", GetShortPluginName());
|
|
|
|
else
|
|
|
|
error.SetErrorStringWithFormat ("Platform::ConnectRemote() is not supported by %s", GetShortPluginName());
|
2011-03-09 06:40:15 +08:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
Error
|
2011-03-23 08:09:55 +08:00
|
|
|
Platform::DisconnectRemote ()
|
2011-03-09 06:40:15 +08:00
|
|
|
{
|
|
|
|
Error error;
|
2011-03-23 08:09:55 +08:00
|
|
|
if (IsHost())
|
|
|
|
error.SetErrorStringWithFormat ("The currently selected platform (%s) is the host platform and is always connected.", GetShortPluginName());
|
|
|
|
else
|
|
|
|
error.SetErrorStringWithFormat ("Platform::DisconnectRemote() is not supported by %s", GetShortPluginName());
|
2011-03-09 06:40:15 +08:00
|
|
|
return error;
|
|
|
|
}
|