blob: 8d907f4326978ac80a08c3899ef0ad58a786bba6 [file] [log] [blame]
Chris Lattner24943d22010-06-08 16:52:24 +00001//===-- TargetList.cpp ------------------------------------------*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Daniel Malead891f9b2012-12-05 00:20:57 +000010#include "lldb/lldb-python.h"
11
Chris Lattner24943d22010-06-08 16:52:24 +000012// C Includes
13// C++ Includes
14// Other libraries and framework includes
15// Project includes
16#include "lldb/Core/Broadcaster.h"
Greg Claytonb1888f22011-03-19 01:12:21 +000017#include "lldb/Core/Debugger.h"
Chris Lattner24943d22010-06-08 16:52:24 +000018#include "lldb/Core/Event.h"
Greg Clayton49ce8962012-08-29 21:13:06 +000019#include "lldb/Core/Module.h"
Greg Clayton36b877d2013-04-24 22:29:28 +000020#include "lldb/Core/ModuleSpec.h"
Chris Lattner24943d22010-06-08 16:52:24 +000021#include "lldb/Core/State.h"
22#include "lldb/Core/Timer.h"
23#include "lldb/Host/Host.h"
Greg Claytonb1db6582012-03-20 18:34:04 +000024#include "lldb/Interpreter/CommandInterpreter.h"
Greg Clayton3e8c25f2011-09-24 00:52:29 +000025#include "lldb/Interpreter/OptionGroupPlatform.h"
Greg Clayton36b877d2013-04-24 22:29:28 +000026#include "lldb/Symbol/ObjectFile.h"
Greg Claytone4b9c1f2011-03-08 22:40:15 +000027#include "lldb/Target/Platform.h"
Chris Lattner24943d22010-06-08 16:52:24 +000028#include "lldb/Target/Process.h"
29#include "lldb/Target/TargetList.h"
30
31using namespace lldb;
32using namespace lldb_private;
33
Jim Ingham5a15e692012-02-16 06:50:00 +000034ConstString &
35TargetList::GetStaticBroadcasterClass ()
36{
37 static ConstString class_name ("lldb.targetList");
38 return class_name;
39}
Chris Lattner24943d22010-06-08 16:52:24 +000040
41//----------------------------------------------------------------------
42// TargetList constructor
43//----------------------------------------------------------------------
Jim Ingham5a15e692012-02-16 06:50:00 +000044TargetList::TargetList(Debugger &debugger) :
Jim Ingham94a5d0d2012-10-10 18:32:14 +000045 Broadcaster(&debugger, TargetList::GetStaticBroadcasterClass().AsCString()),
Chris Lattner24943d22010-06-08 16:52:24 +000046 m_target_list(),
47 m_target_list_mutex (Mutex::eMutexTypeRecursive),
Jim Inghamc8332952010-08-26 21:32:51 +000048 m_selected_target_idx (0)
Chris Lattner24943d22010-06-08 16:52:24 +000049{
Jim Ingham5a15e692012-02-16 06:50:00 +000050 CheckInWithManager();
Chris Lattner24943d22010-06-08 16:52:24 +000051}
52
53//----------------------------------------------------------------------
54// Destructor
55//----------------------------------------------------------------------
56TargetList::~TargetList()
57{
58 Mutex::Locker locker(m_target_list_mutex);
59 m_target_list.clear();
60}
61
62Error
Greg Clayton3e8c25f2011-09-24 00:52:29 +000063TargetList::CreateTarget (Debugger &debugger,
Greg Claytoned0a0fb2012-10-18 16:33:33 +000064 const char *user_exe_path,
Greg Clayton3e8c25f2011-09-24 00:52:29 +000065 const char *triple_cstr,
66 bool get_dependent_files,
67 const OptionGroupPlatform *platform_options,
68 TargetSP &target_sp)
69{
70 Error error;
71 PlatformSP platform_sp;
Greg Clayton3e8c25f2011-09-24 00:52:29 +000072
Johnny Chen94113c42012-01-05 01:26:01 +000073 // This is purposely left empty unless it is specified by triple_cstr.
74 // If not initialized via triple_cstr, then the currently selected platform
75 // will set the architecture correctly.
Greg Claytonb170aee2012-05-08 01:45:38 +000076 const ArchSpec arch(triple_cstr);
77 if (triple_cstr && triple_cstr[0])
Greg Clayton3e8c25f2011-09-24 00:52:29 +000078 {
Greg Clayton3e8c25f2011-09-24 00:52:29 +000079 if (!arch.IsValid())
80 {
Greg Clayton9c236732011-10-26 00:56:27 +000081 error.SetErrorStringWithFormat("invalid triple '%s'", triple_cstr);
Greg Clayton3e8c25f2011-09-24 00:52:29 +000082 return error;
83 }
84 }
Greg Clayton36b877d2013-04-24 22:29:28 +000085
Greg Claytonb170aee2012-05-08 01:45:38 +000086 ArchSpec platform_arch(arch);
Greg Clayton36b877d2013-04-24 22:29:28 +000087
88
89 if (user_exe_path && user_exe_path[0])
90 {
91 ModuleSpecList module_specs;
92 ModuleSpec module_spec;
93 module_spec.GetFileSpec().SetFile(user_exe_path, true);
94 lldb::offset_t file_offset = 0;
Greg Clayton7980d352013-07-12 22:07:46 +000095 lldb::offset_t file_size = 0;
96 const size_t num_specs = ObjectFile::GetModuleSpecifications (module_spec.GetFileSpec(), file_offset, file_size, module_specs);
Greg Clayton36b877d2013-04-24 22:29:28 +000097 if (num_specs > 0)
98 {
99 ModuleSpec matching_module_spec;
100
101 if (num_specs == 1)
102 {
103 if (module_specs.GetModuleSpecAtIndex(0, matching_module_spec))
104 {
105 if (platform_arch.IsValid())
106 {
107 if (!platform_arch.IsCompatibleMatch(matching_module_spec.GetArchitecture()))
108 {
Greg Clayton97a19b22013-04-29 17:25:54 +0000109 error.SetErrorStringWithFormat("the specified architecture '%s' is not compatible with '%s' in '%s'",
Greg Clayton36b877d2013-04-24 22:29:28 +0000110 platform_arch.GetTriple().str().c_str(),
111 matching_module_spec.GetArchitecture().GetTriple().str().c_str(),
Greg Clayton97a19b22013-04-29 17:25:54 +0000112 module_spec.GetFileSpec().GetPath().c_str());
Greg Clayton36b877d2013-04-24 22:29:28 +0000113 return error;
114 }
115 }
116 else
117 {
118 // Only one arch and none was specified
119 platform_arch = matching_module_spec.GetArchitecture();
120 }
121 }
122 }
123 else
124 {
125 if (arch.IsValid())
126 {
127 module_spec.GetArchitecture() = arch;
128 if (module_specs.FindMatchingModuleSpec(module_spec, matching_module_spec))
129 {
130 platform_arch = matching_module_spec.GetArchitecture();
131 }
132 }
Greg Claytonee72ab62013-05-03 23:22:20 +0000133 // Don't just select the first architecture, we want to let the platform select
134 // the best architecture first when there are multiple archs.
135// else
136// {
137// // No arch specified, select the first arch
138// if (module_specs.GetModuleSpecAtIndex(0, matching_module_spec))
139// {
140// platform_arch = matching_module_spec.GetArchitecture();
141// }
142// }
Greg Clayton36b877d2013-04-24 22:29:28 +0000143 }
144 }
145 }
146
Greg Claytonb1db6582012-03-20 18:34:04 +0000147 CommandInterpreter &interpreter = debugger.GetCommandInterpreter();
148 if (platform_options)
149 {
150 if (platform_options->PlatformWasSpecified ())
151 {
152 const bool select_platform = true;
153 platform_sp = platform_options->CreatePlatformWithOptions (interpreter,
154 arch,
155 select_platform,
Greg Claytonb170aee2012-05-08 01:45:38 +0000156 error,
157 platform_arch);
Greg Claytonb1db6582012-03-20 18:34:04 +0000158 if (!platform_sp)
159 return error;
160 }
161 }
162
163 if (!platform_sp)
164 {
165 // Get the current platform and make sure it is compatible with the
166 // current architecture if we have a valid architecture.
167 platform_sp = debugger.GetPlatformList().GetSelectedPlatform ();
168
Greg Claytonaad2b0f2013-01-11 20:49:54 +0000169 if (arch.IsValid() && !platform_sp->IsCompatibleArchitecture(arch, false, &platform_arch))
Greg Claytonb1db6582012-03-20 18:34:04 +0000170 {
Greg Claytonb170aee2012-05-08 01:45:38 +0000171 platform_sp = Platform::GetPlatformForArchitecture(arch, &platform_arch);
Greg Claytonb1db6582012-03-20 18:34:04 +0000172 }
173 }
Greg Claytonb170aee2012-05-08 01:45:38 +0000174
175 if (!platform_arch.IsValid())
176 platform_arch = arch;
Greg Claytonb1db6582012-03-20 18:34:04 +0000177
Greg Clayton3e8c25f2011-09-24 00:52:29 +0000178 error = TargetList::CreateTarget (debugger,
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000179 user_exe_path,
Greg Claytonb170aee2012-05-08 01:45:38 +0000180 platform_arch,
Greg Clayton3e8c25f2011-09-24 00:52:29 +0000181 get_dependent_files,
182 platform_sp,
183 target_sp);
184 return error;
185}
186
187Error
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000188TargetList::CreateTarget (Debugger &debugger,
189 const char *user_exe_path,
190 const ArchSpec& specified_arch,
191 bool get_dependent_files,
192 PlatformSP &platform_sp,
193 TargetSP &target_sp)
Chris Lattner24943d22010-06-08 16:52:24 +0000194{
195 Timer scoped_timer (__PRETTY_FUNCTION__,
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000196 "TargetList::CreateTarget (file = '%s', arch = '%s')",
197 user_exe_path,
Greg Claytonb170aee2012-05-08 01:45:38 +0000198 specified_arch.GetArchitectureName());
Jim Ingham7508e732010-08-09 23:31:02 +0000199 Error error;
Greg Claytonb1888f22011-03-19 01:12:21 +0000200
Greg Claytonb170aee2012-05-08 01:45:38 +0000201 ArchSpec arch(specified_arch);
202
203 if (platform_sp)
204 {
205 if (arch.IsValid())
206 {
Greg Claytonaad2b0f2013-01-11 20:49:54 +0000207 if (!platform_sp->IsCompatibleArchitecture(arch, false, NULL))
Greg Claytonb170aee2012-05-08 01:45:38 +0000208 platform_sp = Platform::GetPlatformForArchitecture(specified_arch, &arch);
209 }
210 }
211 else if (arch.IsValid())
212 {
213 platform_sp = Platform::GetPlatformForArchitecture(specified_arch, &arch);
214 }
Jim Ingham7508e732010-08-09 23:31:02 +0000215
Greg Claytonb170aee2012-05-08 01:45:38 +0000216 if (!platform_sp)
217 platform_sp = debugger.GetPlatformList().GetSelectedPlatform();
218
Greg Clayton7f206ba2012-06-05 21:17:09 +0000219 if (!arch.IsValid())
220 arch = specified_arch;
Greg Clayton7f206ba2012-06-05 21:17:09 +0000221
Jason Molenda6f8d3852013-04-19 22:38:50 +0000222 FileSpec file (user_exe_path, false);
223 if (!file.Exists() && user_exe_path && user_exe_path[0] == '~')
224 {
Michael Sartain4349bcb2013-07-31 23:27:46 +0000225 // we want to expand the tilde but we don't want to resolve any symbolic links
226 // so we can't use the FileSpec constructor's resolve flag
227 char unglobbed_path[PATH_MAX];
228 unglobbed_path[0] = '\0';
229
230 size_t return_count = FileSpec::ResolveUsername(user_exe_path, unglobbed_path, sizeof(unglobbed_path));
231
232 if (return_count == 0 || return_count >= sizeof(unglobbed_path))
233 ::snprintf (unglobbed_path, sizeof(unglobbed_path), "%s", user_exe_path);
234
235 file = FileSpec(unglobbed_path, false);
Jason Molenda6f8d3852013-04-19 22:38:50 +0000236 }
Michael Sartain4349bcb2013-07-31 23:27:46 +0000237
Greg Claytonfb467722012-10-25 22:45:35 +0000238 bool user_exe_path_is_bundle = false;
239 char resolved_bundle_exe_path[PATH_MAX];
240 resolved_bundle_exe_path[0] = '\0';
Greg Claytone4b9c1f2011-03-08 22:40:15 +0000241 if (file)
Jim Ingham7508e732010-08-09 23:31:02 +0000242 {
Greg Claytonfb467722012-10-25 22:45:35 +0000243 if (file.GetFileType() == FileSpec::eFileTypeDirectory)
244 user_exe_path_is_bundle = true;
245
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000246 if (file.IsRelativeToCurrentWorkingDirectory())
247 {
248 // Ignore paths that start with "./" and "../"
249 if (!((user_exe_path[0] == '.' && user_exe_path[1] == '/') ||
250 (user_exe_path[0] == '.' && user_exe_path[1] == '.' && user_exe_path[2] == '/')))
251 {
252 char cwd[PATH_MAX];
253 if (getcwd (cwd, sizeof(cwd)))
254 {
255 std::string cwd_user_exe_path (cwd);
256 cwd_user_exe_path += '/';
257 cwd_user_exe_path += user_exe_path;
Greg Claytonedd6ade2013-04-04 00:15:09 +0000258 FileSpec cwd_file (cwd_user_exe_path.c_str(), false);
259 if (cwd_file.Exists())
260 file = cwd_file;
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000261 }
262 }
263 }
264
Jim Ingham7508e732010-08-09 23:31:02 +0000265 ModuleSP exe_module_sp;
Greg Claytone4b9c1f2011-03-08 22:40:15 +0000266 if (platform_sp)
Greg Clayton9ce95382012-02-13 23:10:39 +0000267 {
268 FileSpecList executable_search_paths (Target::GetDefaultExecutableSearchPaths());
Greg Claytonb170aee2012-05-08 01:45:38 +0000269 error = platform_sp->ResolveExecutable (file,
270 arch,
Greg Clayton9ce95382012-02-13 23:10:39 +0000271 exe_module_sp,
272 executable_search_paths.GetSize() ? &executable_search_paths : NULL);
273 }
Chris Lattner24943d22010-06-08 16:52:24 +0000274
Greg Claytone4b9c1f2011-03-08 22:40:15 +0000275 if (error.Success() && exe_module_sp)
Chris Lattner24943d22010-06-08 16:52:24 +0000276 {
Jim Ingham7508e732010-08-09 23:31:02 +0000277 if (exe_module_sp->GetObjectFile() == NULL)
278 {
Greg Claytonfa2f48b2010-12-08 04:55:11 +0000279 if (arch.IsValid())
280 {
Greg Clayton97a19b22013-04-29 17:25:54 +0000281 error.SetErrorStringWithFormat("\"%s\" doesn't contain architecture %s",
282 file.GetPath().c_str(),
Greg Clayton940b1032011-02-23 00:35:02 +0000283 arch.GetArchitectureName());
Greg Claytonfa2f48b2010-12-08 04:55:11 +0000284 }
285 else
286 {
Greg Clayton97a19b22013-04-29 17:25:54 +0000287 error.SetErrorStringWithFormat("unsupported file type \"%s\"",
288 file.GetPath().c_str());
Greg Claytonfa2f48b2010-12-08 04:55:11 +0000289 }
Jim Ingham7508e732010-08-09 23:31:02 +0000290 return error;
291 }
Greg Clayton24bc5d92011-03-30 18:16:51 +0000292 target_sp.reset(new Target(debugger, arch, platform_sp));
Jim Ingham7508e732010-08-09 23:31:02 +0000293 target_sp->SetExecutableModule (exe_module_sp, get_dependent_files);
Greg Claytonfb467722012-10-25 22:45:35 +0000294 if (user_exe_path_is_bundle)
295 exe_module_sp->GetFileSpec().GetPath(resolved_bundle_exe_path, sizeof(resolved_bundle_exe_path));
Chris Lattner24943d22010-06-08 16:52:24 +0000296 }
Jim Ingham7508e732010-08-09 23:31:02 +0000297 }
Greg Claytone4b9c1f2011-03-08 22:40:15 +0000298 else
Jim Ingham7508e732010-08-09 23:31:02 +0000299 {
Greg Claytone4b9c1f2011-03-08 22:40:15 +0000300 // No file was specified, just create an empty target with any arch
301 // if a valid arch was specified
Greg Clayton24bc5d92011-03-30 18:16:51 +0000302 target_sp.reset(new Target(debugger, arch, platform_sp));
Greg Claytone4b9c1f2011-03-08 22:40:15 +0000303 }
304
305 if (target_sp)
306 {
Greg Claytonfb467722012-10-25 22:45:35 +0000307 // Set argv0 with what the user typed, unless the user specified a
308 // directory. If the user specified a directory, then it is probably a
309 // bundle that was resolved and we need to use the resolved bundle path
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000310 if (user_exe_path)
311 {
312 // Use exactly what the user typed as the first argument when we exec or posix_spawn
Greg Claytonfb467722012-10-25 22:45:35 +0000313 if (user_exe_path_is_bundle && resolved_bundle_exe_path[0])
314 {
315 target_sp->SetArg0 (resolved_bundle_exe_path);
316 }
317 else
318 {
Michael Sartain4349bcb2013-07-31 23:27:46 +0000319 // Use resolved path
320 target_sp->SetArg0 (file.GetPath().c_str());
Greg Claytonfb467722012-10-25 22:45:35 +0000321 }
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000322 }
323 if (file.GetDirectory())
324 {
325 FileSpec file_dir;
326 file_dir.GetDirectory() = file.GetDirectory();
327 target_sp->GetExecutableSearchPaths ().Append (file_dir);
328 }
Jim Ingham7508e732010-08-09 23:31:02 +0000329 Mutex::Locker locker(m_target_list_mutex);
Jim Inghamc8332952010-08-26 21:32:51 +0000330 m_selected_target_idx = m_target_list.size();
Jim Ingham7508e732010-08-09 23:31:02 +0000331 m_target_list.push_back(target_sp);
Greg Claytoned0a0fb2012-10-18 16:33:33 +0000332
333
Jim Ingham7508e732010-08-09 23:31:02 +0000334 }
Chris Lattner24943d22010-06-08 16:52:24 +0000335
Chris Lattner24943d22010-06-08 16:52:24 +0000336 return error;
337}
338
339bool
340TargetList::DeleteTarget (TargetSP &target_sp)
341{
342 Mutex::Locker locker(m_target_list_mutex);
343 collection::iterator pos, end = m_target_list.end();
344
345 for (pos = m_target_list.begin(); pos != end; ++pos)
346 {
347 if (pos->get() == target_sp.get())
348 {
349 m_target_list.erase(pos);
350 return true;
351 }
352 }
353 return false;
354}
355
356
357TargetSP
358TargetList::FindTargetWithExecutableAndArchitecture
359(
360 const FileSpec &exe_file_spec,
361 const ArchSpec *exe_arch_ptr
362) const
363{
364 Mutex::Locker locker (m_target_list_mutex);
365 TargetSP target_sp;
366 bool full_match = exe_file_spec.GetDirectory();
367
368 collection::const_iterator pos, end = m_target_list.end();
369 for (pos = m_target_list.begin(); pos != end; ++pos)
370 {
Greg Clayton5beb99d2011-08-11 02:48:45 +0000371 Module *exe_module = (*pos)->GetExecutableModulePointer();
Chris Lattner24943d22010-06-08 16:52:24 +0000372
Greg Clayton5beb99d2011-08-11 02:48:45 +0000373 if (exe_module)
Chris Lattner24943d22010-06-08 16:52:24 +0000374 {
Greg Clayton5beb99d2011-08-11 02:48:45 +0000375 if (FileSpec::Equal (exe_file_spec, exe_module->GetFileSpec(), full_match))
Chris Lattner24943d22010-06-08 16:52:24 +0000376 {
377 if (exe_arch_ptr)
378 {
Sean Callanan40e278c2012-12-13 22:07:14 +0000379 if (!exe_arch_ptr->IsCompatibleMatch(exe_module->GetArchitecture()))
Chris Lattner24943d22010-06-08 16:52:24 +0000380 continue;
381 }
382 target_sp = *pos;
383 break;
384 }
385 }
386 }
387 return target_sp;
388}
389
390TargetSP
391TargetList::FindTargetWithProcessID (lldb::pid_t pid) const
392{
393 Mutex::Locker locker(m_target_list_mutex);
394 TargetSP target_sp;
395 collection::const_iterator pos, end = m_target_list.end();
396 for (pos = m_target_list.begin(); pos != end; ++pos)
397 {
398 Process* process = (*pos)->GetProcessSP().get();
399 if (process && process->GetID() == pid)
400 {
401 target_sp = *pos;
402 break;
403 }
404 }
405 return target_sp;
406}
407
408
409TargetSP
410TargetList::FindTargetWithProcess (Process *process) const
411{
412 TargetSP target_sp;
413 if (process)
414 {
415 Mutex::Locker locker(m_target_list_mutex);
416 collection::const_iterator pos, end = m_target_list.end();
417 for (pos = m_target_list.begin(); pos != end; ++pos)
418 {
419 if (process == (*pos)->GetProcessSP().get())
420 {
421 target_sp = *pos;
422 break;
423 }
424 }
425 }
426 return target_sp;
427}
428
429TargetSP
430TargetList::GetTargetSP (Target *target) const
431{
432 TargetSP target_sp;
433 if (target)
434 {
435 Mutex::Locker locker(m_target_list_mutex);
436 collection::const_iterator pos, end = m_target_list.end();
437 for (pos = m_target_list.begin(); pos != end; ++pos)
438 {
439 if (target == (*pos).get())
440 {
441 target_sp = *pos;
442 break;
443 }
444 }
445 }
446 return target_sp;
447}
448
449uint32_t
450TargetList::SendAsyncInterrupt (lldb::pid_t pid)
451{
452 uint32_t num_async_interrupts_sent = 0;
453
454 if (pid != LLDB_INVALID_PROCESS_ID)
455 {
456 TargetSP target_sp(FindTargetWithProcessID (pid));
457 if (target_sp.get())
458 {
459 Process* process = target_sp->GetProcessSP().get();
460 if (process)
461 {
Jim Ingham5d90ade2012-07-27 23:57:19 +0000462 process->SendAsyncInterrupt();
Chris Lattner24943d22010-06-08 16:52:24 +0000463 ++num_async_interrupts_sent;
464 }
465 }
466 }
467 else
468 {
469 // We don't have a valid pid to broadcast to, so broadcast to the target
470 // list's async broadcaster...
471 BroadcastEvent (Process::eBroadcastBitInterrupt, NULL);
472 }
473
474 return num_async_interrupts_sent;
475}
476
477uint32_t
478TargetList::SignalIfRunning (lldb::pid_t pid, int signo)
479{
480 uint32_t num_signals_sent = 0;
481 Process *process = NULL;
482 if (pid == LLDB_INVALID_PROCESS_ID)
483 {
484 // Signal all processes with signal
485 Mutex::Locker locker(m_target_list_mutex);
486 collection::iterator pos, end = m_target_list.end();
487 for (pos = m_target_list.begin(); pos != end; ++pos)
488 {
489 process = (*pos)->GetProcessSP().get();
490 if (process)
491 {
492 if (process->IsAlive())
493 {
494 ++num_signals_sent;
495 process->Signal (signo);
496 }
497 }
498 }
499 }
500 else
501 {
502 // Signal a specific process with signal
503 TargetSP target_sp(FindTargetWithProcessID (pid));
504 if (target_sp.get())
505 {
506 process = target_sp->GetProcessSP().get();
507 if (process)
508 {
509 if (process->IsAlive())
510 {
511 ++num_signals_sent;
512 process->Signal (signo);
513 }
514 }
515 }
516 }
517 return num_signals_sent;
518}
519
520int
521TargetList::GetNumTargets () const
522{
523 Mutex::Locker locker (m_target_list_mutex);
524 return m_target_list.size();
525}
526
527lldb::TargetSP
528TargetList::GetTargetAtIndex (uint32_t idx) const
529{
530 TargetSP target_sp;
531 Mutex::Locker locker (m_target_list_mutex);
532 if (idx < m_target_list.size())
533 target_sp = m_target_list[idx];
534 return target_sp;
535}
536
537uint32_t
Jim Inghamf92ddcc2012-05-08 23:06:07 +0000538TargetList::GetIndexOfTarget (lldb::TargetSP target_sp) const
539{
540 Mutex::Locker locker (m_target_list_mutex);
541 size_t num_targets = m_target_list.size();
542 for (size_t idx = 0; idx < num_targets; idx++)
543 {
544 if (target_sp == m_target_list[idx])
545 return idx;
546 }
547 return UINT32_MAX;
548}
549
550uint32_t
Jim Inghamc8332952010-08-26 21:32:51 +0000551TargetList::SetSelectedTarget (Target* target)
Chris Lattner24943d22010-06-08 16:52:24 +0000552{
553 Mutex::Locker locker (m_target_list_mutex);
554 collection::const_iterator pos,
555 begin = m_target_list.begin(),
556 end = m_target_list.end();
557 for (pos = begin; pos != end; ++pos)
558 {
559 if (pos->get() == target)
560 {
Jim Inghamc8332952010-08-26 21:32:51 +0000561 m_selected_target_idx = std::distance (begin, pos);
562 return m_selected_target_idx;
Chris Lattner24943d22010-06-08 16:52:24 +0000563 }
564 }
Jim Inghamc8332952010-08-26 21:32:51 +0000565 m_selected_target_idx = 0;
566 return m_selected_target_idx;
Chris Lattner24943d22010-06-08 16:52:24 +0000567}
568
569lldb::TargetSP
Jim Inghamc8332952010-08-26 21:32:51 +0000570TargetList::GetSelectedTarget ()
Chris Lattner24943d22010-06-08 16:52:24 +0000571{
572 Mutex::Locker locker (m_target_list_mutex);
Jim Inghamc8332952010-08-26 21:32:51 +0000573 if (m_selected_target_idx >= m_target_list.size())
574 m_selected_target_idx = 0;
575 return GetTargetAtIndex (m_selected_target_idx);
Chris Lattner24943d22010-06-08 16:52:24 +0000576}