| /* List of target connections for GDB. | 
 |  | 
 |    Copyright (C) 2017-2023 Free Software Foundation, Inc. | 
 |  | 
 |    This file is part of GDB. | 
 |  | 
 |    This program is free software; you can redistribute it and/or modify | 
 |    it under the terms of the GNU General Public License as published by | 
 |    the Free Software Foundation; either version 3 of the License, or | 
 |    (at your option) any later version. | 
 |  | 
 |    This program is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 |    GNU General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU General Public License | 
 |    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include "defs.h" | 
 | #include "target-connection.h" | 
 |  | 
 | #include <map> | 
 |  | 
 | #include "inferior.h" | 
 | #include "target.h" | 
 | #include "observable.h" | 
 |  | 
 | /* A map between connection number and representative process_stratum | 
 |    target.  */ | 
 | static std::map<int, process_stratum_target *> process_targets; | 
 |  | 
 | /* The highest connection number ever given to a target.  */ | 
 | static int highest_target_connection_num; | 
 |  | 
 | /* See target-connection.h.  */ | 
 |  | 
 | void | 
 | connection_list_add (process_stratum_target *t) | 
 | { | 
 |   if (t->connection_number == 0) | 
 |     { | 
 |       t->connection_number = ++highest_target_connection_num; | 
 |       process_targets[t->connection_number] = t; | 
 |     } | 
 | } | 
 |  | 
 | /* See target-connection.h.  */ | 
 |  | 
 | void | 
 | connection_list_remove (process_stratum_target *t) | 
 | { | 
 |   /* Notify about the connection being removed before we reset the | 
 |      connection number to zero.  */ | 
 |   gdb::observers::connection_removed.notify (t); | 
 |   process_targets.erase (t->connection_number); | 
 |   t->connection_number = 0; | 
 | } | 
 |  | 
 | /* See target-connection.h.  */ | 
 |  | 
 | std::string | 
 | make_target_connection_string (process_stratum_target *t) | 
 | { | 
 |   if (t->connection_string () != NULL) | 
 |     return string_printf ("%s %s", t->shortname (), | 
 | 			  t->connection_string ()); | 
 |   else | 
 |     return t->shortname (); | 
 | } | 
 |  | 
 | /* Prints the list of target connections and their details on UIOUT. | 
 |  | 
 |    If REQUESTED_CONNECTIONS is not NULL, it's a list of GDB ids of the | 
 |    target connections that should be printed.  Otherwise, all target | 
 |    connections are printed.  */ | 
 |  | 
 | static void | 
 | print_connection (struct ui_out *uiout, const char *requested_connections) | 
 | { | 
 |   int count = 0; | 
 |   size_t what_len = 0; | 
 |  | 
 |   /* Compute number of lines we will print.  */ | 
 |   for (const auto &it : process_targets) | 
 |     { | 
 |       if (!number_is_in_list (requested_connections, it.first)) | 
 | 	continue; | 
 |  | 
 |       ++count; | 
 |  | 
 |       process_stratum_target *t = it.second; | 
 |  | 
 |       size_t l = make_target_connection_string (t).length (); | 
 |       if (l > what_len) | 
 | 	what_len = l; | 
 |     } | 
 |  | 
 |   if (count == 0) | 
 |     { | 
 |       uiout->message (_("No connections.\n")); | 
 |       return; | 
 |     } | 
 |  | 
 |   ui_out_emit_table table_emitter (uiout, 4, process_targets.size (), | 
 | 				   "connections"); | 
 |  | 
 |   uiout->table_header (1, ui_left, "current", ""); | 
 |   uiout->table_header (4, ui_left, "number", "Num"); | 
 |   /* The text in the "what" column may include spaces.  Add one extra | 
 |      space to visually separate the What and Description columns a | 
 |      little better.  Compare: | 
 |       "* 1    remote :9999 Remote serial target in gdb-specific protocol" | 
 |       "* 1    remote :9999  Remote serial target in gdb-specific protocol" | 
 |   */ | 
 |   uiout->table_header (what_len + 1, ui_left, "what", "What"); | 
 |   uiout->table_header (17, ui_left, "description", "Description"); | 
 |  | 
 |   uiout->table_body (); | 
 |  | 
 |   for (const auto &it : process_targets) | 
 |     { | 
 |       process_stratum_target *t = it.second; | 
 |  | 
 |       if (!number_is_in_list (requested_connections, t->connection_number)) | 
 | 	continue; | 
 |  | 
 |       ui_out_emit_tuple tuple_emitter (uiout, NULL); | 
 |  | 
 |       if (current_inferior ()->process_target () == t) | 
 | 	uiout->field_string ("current", "*"); | 
 |       else | 
 | 	uiout->field_skip ("current"); | 
 |  | 
 |       uiout->field_signed ("number", t->connection_number); | 
 |  | 
 |       uiout->field_string ("what", make_target_connection_string (t)); | 
 |  | 
 |       uiout->field_string ("description", t->longname ()); | 
 |  | 
 |       uiout->text ("\n"); | 
 |     } | 
 | } | 
 |  | 
 | /* The "info connections" command.  */ | 
 |  | 
 | static void | 
 | info_connections_command (const char *args, int from_tty) | 
 | { | 
 |   print_connection (current_uiout, args); | 
 | } | 
 |  | 
 | void _initialize_target_connection (); | 
 |  | 
 | void | 
 | _initialize_target_connection () | 
 | { | 
 |   add_info ("connections", info_connections_command, | 
 | 	    _("\ | 
 | Target connections in use.\n\ | 
 | Shows the list of target connections currently in use.")); | 
 | } |