gdb/nat/linux-osdata.c - gdb

Global variables defined

Data types defined

Functions defined

Macros defined

Source code

  1. /* Linux-specific functions to retrieve OS data.

  2.    Copyright (C) 2009-2015 Free Software Foundation, Inc.

  3.    This file is part of GDB.

  4.    This program is free software; you can redistribute it and/or modify
  5.    it under the terms of the GNU General Public License as published by
  6.    the Free Software Foundation; either version 3 of the License, or
  7.    (at your option) any later version.

  8.    This program is distributed in the hope that it will be useful,
  9.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  10.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  11.    GNU General Public License for more details.

  12.    You should have received a copy of the GNU General Public License
  13.    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */

  14. #include "common-defs.h"
  15. #include "linux-osdata.h"

  16. #include <sys/types.h>
  17. #include <sys/sysinfo.h>
  18. #include <ctype.h>
  19. #include <utmp.h>
  20. #include <time.h>
  21. #include <unistd.h>
  22. #include <pwd.h>
  23. #include <grp.h>
  24. #include <netdb.h>
  25. #include <netinet/in.h>
  26. #include <arpa/inet.h>

  27. #include "xml-utils.h"
  28. #include "buffer.h"
  29. #include <dirent.h>
  30. #include <sys/stat.h>
  31. #include "filestuff.h"

  32. #define NAMELEN(dirent) strlen ((dirent)->d_name)

  33. /* Define PID_T to be a fixed size that is at least as large as pid_t,
  34.    so that reading pid values embedded in /proc works
  35.    consistently.  */

  36. typedef long long  PID_T;

  37. /* Define TIME_T to be at least as large as time_t, so that reading
  38.    time values embedded in /proc works consistently.  */

  39. typedef long long TIME_T;

  40. #define MAX_PID_T_STRLEN  (sizeof ("-9223372036854775808") - 1)

  41. /* Returns the CPU core that thread PTID is currently running on.  */

  42. /* Compute and return the processor core of a given thread.  */

  43. int
  44. linux_common_core_of_thread (ptid_t ptid)
  45. {
  46.   char filename[sizeof ("/proc//task//stat") + 2 * MAX_PID_T_STRLEN];
  47.   FILE *f;
  48.   char *content = NULL;
  49.   char *p;
  50.   char *ts = 0;
  51.   int content_read = 0;
  52.   int i;
  53.   int core;

  54.   sprintf (filename, "/proc/%lld/task/%lld/stat",
  55.            (PID_T) ptid_get_pid (ptid), (PID_T) ptid_get_lwp (ptid));
  56.   f = gdb_fopen_cloexec (filename, "r");
  57.   if (!f)
  58.     return -1;

  59.   for (;;)
  60.     {
  61.       int n;
  62.       content = xrealloc (content, content_read + 1024);
  63.       n = fread (content + content_read, 1, 1024, f);
  64.       content_read += n;
  65.       if (n < 1024)
  66.         {
  67.           content[content_read] = '\0';
  68.           break;
  69.         }
  70.     }

  71.   /* ps command also relies on no trailing fields ever contain ')'.  */
  72.   p = strrchr (content, ')');
  73.   if (p != NULL)
  74.     p++;

  75.   /* If the first field after program name has index 0, then core number is
  76.      the field with index 36.  There's no constant for that anywhere.  */
  77.   if (p != NULL)
  78.     p = strtok_r (p, " ", &ts);
  79.   for (i = 0; p != NULL && i != 36; ++i)
  80.     p = strtok_r (NULL, " ", &ts);

  81.   if (p == NULL || sscanf (p, "%d", &core) == 0)
  82.     core = -1;

  83.   xfree (content);
  84.   fclose (f);

  85.   return core;
  86. }

  87. /* Finds the command-line of process PID and copies it into COMMAND.
  88.    At most MAXLEN characters are copied.  If the command-line cannot
  89.    be found, PID is copied into command in text-form.  */

  90. static void
  91. command_from_pid (char *command, int maxlen, PID_T pid)
  92. {
  93.   char *stat_path = xstrprintf ("/proc/%lld/stat", pid);
  94.   FILE *fp = gdb_fopen_cloexec (stat_path, "r");

  95.   command[0] = '\0';

  96.   if (fp)
  97.     {
  98.       /* sizeof (cmd) should be greater or equal to TASK_COMM_LEN (in
  99.          include/linux/sched.h in the Linux kernel sources) plus two
  100.          (for the brackets).  */
  101.       char cmd[18];
  102.       PID_T stat_pid;
  103.       int items_read = fscanf (fp, "%lld %17s", &stat_pid, cmd);

  104.       if (items_read == 2 && pid == stat_pid)
  105.         {
  106.           cmd[strlen (cmd) - 1] = '\0'; /* Remove trailing parenthesis.  */
  107.           strncpy (command, cmd + 1, maxlen); /* Ignore leading parenthesis.  */
  108.         }

  109.       fclose (fp);
  110.     }
  111.   else
  112.     {
  113.       /* Return the PID if a /proc entry for the process cannot be found.  */
  114.       snprintf (command, maxlen, "%lld", pid);
  115.     }

  116.   command[maxlen - 1] = '\0'; /* Ensure string is null-terminated.  */

  117.   xfree (stat_path);
  118. }

  119. /* Returns the command-line of the process with the given PID. The
  120.    returned string needs to be freed using xfree after use.  */

  121. static char *
  122. commandline_from_pid (PID_T pid)
  123. {
  124.   char *pathname = xstrprintf ("/proc/%lld/cmdline", pid);
  125.   char *commandline = NULL;
  126.   FILE *f = gdb_fopen_cloexec (pathname, "r");

  127.   if (f)
  128.     {
  129.       size_t len = 0;

  130.       while (!feof (f))
  131.         {
  132.           char buf[1024];
  133.           size_t read_bytes = fread (buf, 1, sizeof (buf), f);

  134.           if (read_bytes)
  135.             {
  136.               commandline = (char *) xrealloc (commandline, len + read_bytes + 1);
  137.               memcpy (commandline + len, buf, read_bytes);
  138.               len += read_bytes;
  139.             }
  140.         }

  141.       fclose (f);

  142.       if (commandline)
  143.         {
  144.           size_t i;

  145.           /* Replace null characters with spaces.  */
  146.           for (i = 0; i < len; ++i)
  147.             if (commandline[i] == '\0')
  148.               commandline[i] = ' ';

  149.           commandline[len] = '\0';
  150.         }
  151.       else
  152.         {
  153.           /* Return the command in square brackets if the command-line
  154.              is empty.  */
  155.           commandline = (char *) xmalloc (32);
  156.           commandline[0] = '[';
  157.           command_from_pid (commandline + 1, 31, pid);

  158.           len = strlen (commandline);
  159.           if (len < 31)
  160.             strcat (commandline, "]");
  161.         }
  162.     }

  163.   xfree (pathname);

  164.   return commandline;
  165. }

  166. /* Finds the user name for the user UID and copies it into USER.  At
  167.    most MAXLEN characters are copied.  */

  168. static void
  169. user_from_uid (char *user, int maxlen, uid_t uid)
  170. {
  171.   struct passwd *pwentry = getpwuid (uid);

  172.   if (pwentry)
  173.     {
  174.       strncpy (user, pwentry->pw_name, maxlen);
  175.       /* Ensure that the user name is null-terminated.  */
  176.       user[maxlen - 1] = '\0';
  177.     }
  178.   else
  179.     user[0] = '\0';
  180. }

  181. /* Finds the owner of process PID and returns the user id in OWNER.
  182.    Returns 0 if the owner was found, -1 otherwise.  */

  183. static int
  184. get_process_owner (uid_t *owner, PID_T pid)
  185. {
  186.   struct stat statbuf;
  187.   char procentry[sizeof ("/proc/") + MAX_PID_T_STRLEN];

  188.   sprintf (procentry, "/proc/%lld", pid);

  189.   if (stat (procentry, &statbuf) == 0 && S_ISDIR (statbuf.st_mode))
  190.     {
  191.       *owner = statbuf.st_uid;
  192.       return 0;
  193.     }
  194.   else
  195.     return -1;
  196. }

  197. /* Find the CPU cores used by process PID and return them in CORES.
  198.    CORES points to an array of NUM_CORES elements.  */

  199. static int
  200. get_cores_used_by_process (PID_T pid, int *cores, const int num_cores)
  201. {
  202.   char taskdir[sizeof ("/proc/") + MAX_PID_T_STRLEN + sizeof ("/task") - 1];
  203.   DIR *dir;
  204.   struct dirent *dp;
  205.   int task_count = 0;

  206.   sprintf (taskdir, "/proc/%lld/task", pid);
  207.   dir = opendir (taskdir);
  208.   if (dir)
  209.     {
  210.       while ((dp = readdir (dir)) != NULL)
  211.         {
  212.           PID_T tid;
  213.           int core;

  214.           if (!isdigit (dp->d_name[0])
  215.               || NAMELEN (dp) > MAX_PID_T_STRLEN)
  216.             continue;

  217.           sscanf (dp->d_name, "%lld", &tid);
  218.           core = linux_common_core_of_thread (ptid_build ((pid_t) pid,
  219.                                                           (pid_t) tid, 0));

  220.           if (core >= 0 && core < num_cores)
  221.             {
  222.               ++cores[core];
  223.               ++task_count;
  224.             }
  225.         }

  226.       closedir (dir);
  227.     }

  228.   return task_count;
  229. }

  230. static LONGEST
  231. linux_xfer_osdata_processes (gdb_byte *readbuf,
  232.                              ULONGEST offset, ULONGEST len)
  233. {
  234.   /* We make the process list snapshot when the object starts to be read.  */
  235.   static const char *buf;
  236.   static LONGEST len_avail = -1;
  237.   static struct buffer buffer;

  238.   if (offset == 0)
  239.     {
  240.       DIR *dirp;

  241.       if (len_avail != -1 && len_avail != 0)
  242.         buffer_free (&buffer);
  243.       len_avail = 0;
  244.       buf = NULL;
  245.       buffer_init (&buffer);
  246.       buffer_grow_str (&buffer, "<osdata type=\"processes\">\n");

  247.       dirp = opendir ("/proc");
  248.       if (dirp)
  249.         {
  250.           const int num_cores = sysconf (_SC_NPROCESSORS_ONLN);
  251.           struct dirent *dp;

  252.           while ((dp = readdir (dirp)) != NULL)
  253.             {
  254.               PID_T pid;
  255.               uid_t owner;
  256.               char user[UT_NAMESIZE];
  257.               char *command_line;
  258.               int *cores;
  259.               int task_count;
  260.               char *cores_str;
  261.               int i;

  262.               if (!isdigit (dp->d_name[0])
  263.                   || NAMELEN (dp) > MAX_PID_T_STRLEN)
  264.                 continue;

  265.               sscanf (dp->d_name, "%lld", &pid);
  266.               command_line = commandline_from_pid (pid);

  267.               if (get_process_owner (&owner, pid) == 0)
  268.                 user_from_uid (user, sizeof (user), owner);
  269.               else
  270.                 strcpy (user, "?");

  271.               /* Find CPU cores used by the process.  */
  272.               cores = (int *) xcalloc (num_cores, sizeof (int));
  273.               task_count = get_cores_used_by_process (pid, cores, num_cores);
  274.               cores_str = (char *) xcalloc (task_count, sizeof ("4294967295") + 1);

  275.               for (i = 0; i < num_cores && task_count > 0; ++i)
  276.                 if (cores[i])
  277.                   {
  278.                     char core_str[sizeof ("4294967295")];

  279.                     sprintf (core_str, "%d", i);
  280.                     strcat (cores_str, core_str);

  281.                     task_count -= cores[i];
  282.                     if (task_count > 0)
  283.                       strcat (cores_str, ",");
  284.                   }

  285.               xfree (cores);

  286.               buffer_xml_printf (
  287.                   &buffer,
  288.                   "<item>"
  289.                   "<column name=\"pid\">%lld</column>"
  290.                   "<column name=\"user\">%s</column>"
  291.                   "<column name=\"command\">%s</column>"
  292.                   "<column name=\"cores\">%s</column>"
  293.                   "</item>",
  294.                   pid,
  295.                   user,
  296.                   command_line ? command_line : "",
  297.                   cores_str);

  298.               xfree (command_line);
  299.               xfree (cores_str);
  300.             }

  301.           closedir (dirp);
  302.         }

  303.       buffer_grow_str0 (&buffer, "</osdata>\n");
  304.       buf = buffer_finish (&buffer);
  305.       len_avail = strlen (buf);
  306.     }

  307.   if (offset >= len_avail)
  308.     {
  309.       /* Done.  Get rid of the buffer.  */
  310.       buffer_free (&buffer);
  311.       buf = NULL;
  312.       len_avail = 0;
  313.       return 0;
  314.     }

  315.   if (len > len_avail - offset)
  316.     len = len_avail - offset;
  317.   memcpy (readbuf, buf + offset, len);

  318.   return len;
  319. }

  320. /* Auxiliary function used by qsort to sort processes by process
  321.    group.  Compares two processes with ids PROCESS1 and PROCESS2.
  322.    PROCESS1 comes before PROCESS2 if it has a lower process group id.
  323.    If they belong to the same process group, PROCESS1 comes before
  324.    PROCESS2 if it has a lower process id or is the process group
  325.    leader.  */

  326. static int
  327. compare_processes (const void *process1, const void *process2)
  328. {
  329.   PID_T pid1 = *((PID_T *) process1);
  330.   PID_T pid2 = *((PID_T *) process2);
  331.   PID_T pgid1 = *((PID_T *) process1 + 1);
  332.   PID_T pgid2 = *((PID_T *) process2 + 1);

  333.   /* Sort by PGID.  */
  334.   if (pgid1 < pgid2)
  335.     return -1;
  336.   else if (pgid1 > pgid2)
  337.     return 1;
  338.   else
  339.     {
  340.       /* Process group leaders always come first, else sort by PID.  */
  341.       if (pid1 == pgid1)
  342.         return -1;
  343.       else if (pid2 == pgid2)
  344.         return 1;
  345.       else if (pid1 < pid2)
  346.         return -1;
  347.       else if (pid1 > pid2)
  348.         return 1;
  349.       else
  350.         return 0;
  351.     }
  352. }

  353. /* Collect all process groups from /proc.  */

  354. static LONGEST
  355. linux_xfer_osdata_processgroups (gdb_byte *readbuf,
  356.                                  ULONGEST offset, ULONGEST len)
  357. {
  358.   /* We make the process list snapshot when the object starts to be read.  */
  359.   static const char *buf;
  360.   static LONGEST len_avail = -1;
  361.   static struct buffer buffer;

  362.   if (offset == 0)
  363.     {
  364.       DIR *dirp;

  365.       if (len_avail != -1 && len_avail != 0)
  366.         buffer_free (&buffer);
  367.       len_avail = 0;
  368.       buf = NULL;
  369.       buffer_init (&buffer);
  370.       buffer_grow_str (&buffer, "<osdata type=\"process groups\">\n");

  371.       dirp = opendir ("/proc");
  372.       if (dirp)
  373.         {
  374.           struct dirent *dp;
  375.           const size_t list_block_size = 512;
  376.           PID_T *process_list = (PID_T *) xmalloc (list_block_size * 2 * sizeof (PID_T));
  377.           size_t process_count = 0;
  378.           size_t i;

  379.           /* Build list consisting of PIDs followed by their
  380.              associated PGID.  */
  381.           while ((dp = readdir (dirp)) != NULL)
  382.             {
  383.               PID_T pid, pgid;

  384.               if (!isdigit (dp->d_name[0])
  385.                   || NAMELEN (dp) > MAX_PID_T_STRLEN)
  386.                 continue;

  387.               sscanf (dp->d_name, "%lld", &pid);
  388.               pgid = getpgid (pid);

  389.               if (pgid > 0)
  390.                 {
  391.                   process_list[2 * process_count] = pid;
  392.                   process_list[2 * process_count + 1] = pgid;
  393.                   ++process_count;

  394.                   /* Increase the size of the list if necessary.  */
  395.                   if (process_count % list_block_size == 0)
  396.                     process_list = (PID_T *) xrealloc (
  397.                         process_list,
  398.                         (process_count + list_block_size)
  399.                         * 2 * sizeof (PID_T));
  400.                 }
  401.             }

  402.           closedir (dirp);

  403.           /* Sort the process list.  */
  404.           qsort (process_list, process_count, 2 * sizeof (PID_T),
  405.                  compare_processes);

  406.           for (i = 0; i < process_count; ++i)
  407.             {
  408.               PID_T pid = process_list[2 * i];
  409.               PID_T pgid = process_list[2 * i + 1];
  410.               char leader_command[32];
  411.               char *command_line;

  412.               command_from_pid (leader_command, sizeof (leader_command), pgid);
  413.               command_line = commandline_from_pid (pid);

  414.               buffer_xml_printf (
  415.                   &buffer,
  416.                   "<item>"
  417.                   "<column name=\"pgid\">%lld</column>"
  418.                   "<column name=\"leader command\">%s</column>"
  419.                   "<column name=\"pid\">%lld</column>"
  420.                   "<column name=\"command line\">%s</column>"
  421.                   "</item>",
  422.                   pgid,
  423.                   leader_command,
  424.                   pid,
  425.                   command_line ? command_line : "");

  426.               xfree (command_line);
  427.             }

  428.           xfree (process_list);
  429.         }

  430.       buffer_grow_str0 (&buffer, "</osdata>\n");
  431.       buf = buffer_finish (&buffer);
  432.       len_avail = strlen (buf);
  433.     }

  434.   if (offset >= len_avail)
  435.     {
  436.       /* Done.  Get rid of the buffer.  */
  437.       buffer_free (&buffer);
  438.       buf = NULL;
  439.       len_avail = 0;
  440.       return 0;
  441.     }

  442.   if (len > len_avail - offset)
  443.     len = len_avail - offset;
  444.   memcpy (readbuf, buf + offset, len);

  445.   return len;
  446. }

  447. /* Collect all the threads in /proc by iterating through processes and
  448.    then tasks within each process.  */

  449. static LONGEST
  450. linux_xfer_osdata_threads (gdb_byte *readbuf,
  451.                            ULONGEST offset, ULONGEST len)
  452. {
  453.   /* We make the process list snapshot when the object starts to be read.  */
  454.   static const char *buf;
  455.   static LONGEST len_avail = -1;
  456.   static struct buffer buffer;

  457.   if (offset == 0)
  458.     {
  459.       DIR *dirp;

  460.       if (len_avail != -1 && len_avail != 0)
  461.         buffer_free (&buffer);
  462.       len_avail = 0;
  463.       buf = NULL;
  464.       buffer_init (&buffer);
  465.       buffer_grow_str (&buffer, "<osdata type=\"threads\">\n");

  466.       dirp = opendir ("/proc");
  467.       if (dirp)
  468.         {
  469.           struct dirent *dp;

  470.           while ((dp = readdir (dirp)) != NULL)
  471.             {
  472.               struct stat statbuf;
  473.               char procentry[sizeof ("/proc/4294967295")];

  474.               if (!isdigit (dp->d_name[0])
  475.                   || NAMELEN (dp) > sizeof ("4294967295") - 1)
  476.                 continue;

  477.               sprintf (procentry, "/proc/%s", dp->d_name);
  478.               if (stat (procentry, &statbuf) == 0
  479.                   && S_ISDIR (statbuf.st_mode))
  480.                 {
  481.                   DIR *dirp2;
  482.                   char *pathname;
  483.                   PID_T pid;
  484.                   char command[32];

  485.                   pathname = xstrprintf ("/proc/%s/task", dp->d_name);

  486.                   pid = atoi (dp->d_name);
  487.                   command_from_pid (command, sizeof (command), pid);

  488.                   dirp2 = opendir (pathname);

  489.                   if (dirp2)
  490.                     {
  491.                       struct dirent *dp2;

  492.                       while ((dp2 = readdir (dirp2)) != NULL)
  493.                         {
  494.                           PID_T tid;
  495.                           int core;

  496.                           if (!isdigit (dp2->d_name[0])
  497.                               || NAMELEN (dp2) > sizeof ("4294967295") - 1)
  498.                             continue;

  499.                           tid = atoi (dp2->d_name);
  500.                           core = linux_common_core_of_thread (ptid_build (pid, tid, 0));

  501.                           buffer_xml_printf (
  502.                             &buffer,
  503.                             "<item>"
  504.                             "<column name=\"pid\">%lld</column>"
  505.                             "<column name=\"command\">%s</column>"
  506.                             "<column name=\"tid\">%lld</column>"
  507.                             "<column name=\"core\">%d</column>"
  508.                             "</item>",
  509.                             pid,
  510.                             command,
  511.                             tid,
  512.                             core);
  513.                         }

  514.                       closedir (dirp2);
  515.                     }

  516.                   xfree (pathname);
  517.                 }
  518.             }

  519.           closedir (dirp);
  520.         }

  521.       buffer_grow_str0 (&buffer, "</osdata>\n");
  522.       buf = buffer_finish (&buffer);
  523.       len_avail = strlen (buf);
  524.     }

  525.   if (offset >= len_avail)
  526.     {
  527.       /* Done.  Get rid of the buffer.  */
  528.       buffer_free (&buffer);
  529.       buf = NULL;
  530.       len_avail = 0;
  531.       return 0;
  532.     }

  533.   if (len > len_avail - offset)
  534.     len = len_avail - offset;
  535.   memcpy (readbuf, buf + offset, len);

  536.   return len;
  537. }

  538. /* Collect all the open file descriptors found in /proc and put the details
  539.    found about them into READBUF.  */

  540. static LONGEST
  541. linux_xfer_osdata_fds (gdb_byte *readbuf,
  542.                        ULONGEST offset, ULONGEST len)
  543. {
  544.   /* We make the process list snapshot when the object starts to be read.  */
  545.   static const char *buf;
  546.   static LONGEST len_avail = -1;
  547.   static struct buffer buffer;

  548.   if (offset == 0)
  549.     {
  550.       DIR *dirp;

  551.       if (len_avail != -1 && len_avail != 0)
  552.         buffer_free (&buffer);
  553.       len_avail = 0;
  554.       buf = NULL;
  555.       buffer_init (&buffer);
  556.       buffer_grow_str (&buffer, "<osdata type=\"files\">\n");

  557.       dirp = opendir ("/proc");
  558.       if (dirp)
  559.         {
  560.           struct dirent *dp;

  561.           while ((dp = readdir (dirp)) != NULL)
  562.             {
  563.               struct stat statbuf;
  564.               char procentry[sizeof ("/proc/4294967295")];

  565.               if (!isdigit (dp->d_name[0])
  566.                   || NAMELEN (dp) > sizeof ("4294967295") - 1)
  567.                 continue;

  568.               sprintf (procentry, "/proc/%s", dp->d_name);
  569.               if (stat (procentry, &statbuf) == 0
  570.                   && S_ISDIR (statbuf.st_mode))
  571.                 {
  572.                   char *pathname;
  573.                   DIR *dirp2;
  574.                   PID_T pid;
  575.                   char command[32];

  576.                   pid = atoi (dp->d_name);
  577.                   command_from_pid (command, sizeof (command), pid);

  578.                   pathname = xstrprintf ("/proc/%s/fd", dp->d_name);
  579.                   dirp2 = opendir (pathname);

  580.                   if (dirp2)
  581.                     {
  582.                       struct dirent *dp2;

  583.                       while ((dp2 = readdir (dirp2)) != NULL)
  584.                         {
  585.                           char *fdname;
  586.                           char buf[1000];
  587.                           ssize_t rslt;

  588.                           if (!isdigit (dp2->d_name[0]))
  589.                             continue;

  590.                           fdname = xstrprintf ("%s/%s", pathname, dp2->d_name);
  591.                           rslt = readlink (fdname, buf, sizeof (buf) - 1);
  592.                           if (rslt >= 0)
  593.                             buf[rslt] = '\0';

  594.                           buffer_xml_printf (
  595.                             &buffer,
  596.                             "<item>"
  597.                             "<column name=\"pid\">%s</column>"
  598.                             "<column name=\"command\">%s</column>"
  599.                             "<column name=\"file descriptor\">%s</column>"
  600.                             "<column name=\"name\">%s</column>"
  601.                             "</item>",
  602.                             dp->d_name,
  603.                             command,
  604.                             dp2->d_name,
  605.                             (rslt >= 0 ? buf : dp2->d_name));
  606.                         }

  607.                       closedir (dirp2);
  608.                     }

  609.                   xfree (pathname);
  610.                 }
  611.             }

  612.           closedir (dirp);
  613.         }

  614.       buffer_grow_str0 (&buffer, "</osdata>\n");
  615.       buf = buffer_finish (&buffer);
  616.       len_avail = strlen (buf);
  617.     }

  618.   if (offset >= len_avail)
  619.     {
  620.       /* Done.  Get rid of the buffer.  */
  621.       buffer_free (&buffer);
  622.       buf = NULL;
  623.       len_avail = 0;
  624.       return 0;
  625.     }

  626.   if (len > len_avail - offset)
  627.     len = len_avail - offset;
  628.   memcpy (readbuf, buf + offset, len);

  629.   return len;
  630. }

  631. /* Returns the socket state STATE in textual form.  */

  632. static const char *
  633. format_socket_state (unsigned char state)
  634. {
  635.   /* Copied from include/net/tcp_states.h in the Linux kernel sources.  */
  636.   enum {
  637.     TCP_ESTABLISHED = 1,
  638.     TCP_SYN_SENT,
  639.     TCP_SYN_RECV,
  640.     TCP_FIN_WAIT1,
  641.     TCP_FIN_WAIT2,
  642.     TCP_TIME_WAIT,
  643.     TCP_CLOSE,
  644.     TCP_CLOSE_WAIT,
  645.     TCP_LAST_ACK,
  646.     TCP_LISTEN,
  647.     TCP_CLOSING
  648.   };

  649.   switch (state)
  650.     {
  651.     case TCP_ESTABLISHED:
  652.       return "ESTABLISHED";
  653.     case TCP_SYN_SENT:
  654.       return "SYN_SENT";
  655.     case TCP_SYN_RECV:
  656.       return "SYN_RECV";
  657.     case TCP_FIN_WAIT1:
  658.       return "FIN_WAIT1";
  659.     case TCP_FIN_WAIT2:
  660.       return "FIN_WAIT2";
  661.     case TCP_TIME_WAIT:
  662.       return "TIME_WAIT";
  663.     case TCP_CLOSE:
  664.       return "CLOSE";
  665.     case TCP_CLOSE_WAIT:
  666.       return "CLOSE_WAIT";
  667.     case TCP_LAST_ACK:
  668.       return "LAST_ACK";
  669.     case TCP_LISTEN:
  670.       return "LISTEN";
  671.     case TCP_CLOSING:
  672.       return "CLOSING";
  673.     default:
  674.       return "(unknown)";
  675.     }
  676. }

  677. union socket_addr
  678.   {
  679.     struct sockaddr sa;
  680.     struct sockaddr_in sin;
  681.     struct sockaddr_in6 sin6;
  682.   };

  683. /* Auxiliary function used by linux_xfer_osdata_isocket.  Formats
  684.    information for all open internet sockets of type FAMILY on the
  685.    system into BUFFER.  If TCP is set, only TCP sockets are processed,
  686.    otherwise only UDP sockets are processed.  */

  687. static void
  688. print_sockets (unsigned short family, int tcp, struct buffer *buffer)
  689. {
  690.   const char *proc_file;
  691.   FILE *fp;

  692.   if (family == AF_INET)
  693.     proc_file = tcp ? "/proc/net/tcp" : "/proc/net/udp";
  694.   else if (family == AF_INET6)
  695.     proc_file = tcp ? "/proc/net/tcp6" : "/proc/net/udp6";
  696.   else
  697.     return;

  698.   fp = gdb_fopen_cloexec (proc_file, "r");
  699.   if (fp)
  700.     {
  701.       char buf[8192];

  702.       do
  703.         {
  704.           if (fgets (buf, sizeof (buf), fp))
  705.             {
  706.               uid_t uid;
  707.               unsigned int local_port, remote_port, state;
  708.               char local_address[NI_MAXHOST], remote_address[NI_MAXHOST];
  709.               int result;

  710. #if NI_MAXHOST <= 32
  711. #error "local_address and remote_address buffers too small"
  712. #endif

  713.               result = sscanf (buf,
  714.                                "%*d: %32[0-9A-F]:%X %32[0-9A-F]:%X %X %*X:%*X %*X:%*X %*X %d %*d %*u %*s\n",
  715.                                local_address, &local_port,
  716.                                remote_address, &remote_port,
  717.                                &state,
  718.                                &uid);

  719.               if (result == 6)
  720.                 {
  721.                   union socket_addr locaddr, remaddr;
  722.                   size_t addr_size;
  723.                   char user[UT_NAMESIZE];
  724.                   char local_service[NI_MAXSERV], remote_service[NI_MAXSERV];

  725.                   if (family == AF_INET)
  726.                     {
  727.                       sscanf (local_address, "%X",
  728.                               &locaddr.sin.sin_addr.s_addr);
  729.                       sscanf (remote_address, "%X",
  730.                               &remaddr.sin.sin_addr.s_addr);

  731.                       locaddr.sin.sin_port = htons (local_port);
  732.                       remaddr.sin.sin_port = htons (remote_port);

  733.                       addr_size = sizeof (struct sockaddr_in);
  734.                     }
  735.                   else
  736.                     {
  737.                       sscanf (local_address, "%8X%8X%8X%8X",
  738.                               locaddr.sin6.sin6_addr.s6_addr32,
  739.                               locaddr.sin6.sin6_addr.s6_addr32 + 1,
  740.                               locaddr.sin6.sin6_addr.s6_addr32 + 2,
  741.                               locaddr.sin6.sin6_addr.s6_addr32 + 3);
  742.                       sscanf (remote_address, "%8X%8X%8X%8X",
  743.                               remaddr.sin6.sin6_addr.s6_addr32,
  744.                               remaddr.sin6.sin6_addr.s6_addr32 + 1,
  745.                               remaddr.sin6.sin6_addr.s6_addr32 + 2,
  746.                               remaddr.sin6.sin6_addr.s6_addr32 + 3);

  747.                       locaddr.sin6.sin6_port = htons (local_port);
  748.                       remaddr.sin6.sin6_port = htons (remote_port);

  749.                       locaddr.sin6.sin6_flowinfo = 0;
  750.                       remaddr.sin6.sin6_flowinfo = 0;
  751.                       locaddr.sin6.sin6_scope_id = 0;
  752.                       remaddr.sin6.sin6_scope_id = 0;

  753.                       addr_size = sizeof (struct sockaddr_in6);
  754.                     }

  755.                   locaddr.sa.sa_family = remaddr.sa.sa_family = family;

  756.                   result = getnameinfo (&locaddr.sa, addr_size,
  757.                                         local_address, sizeof (local_address),
  758.                                         local_service, sizeof (local_service),
  759.                                         NI_NUMERICHOST | NI_NUMERICSERV
  760.                                         | (tcp ? 0 : NI_DGRAM));
  761.                   if (result)
  762.                     continue;

  763.                   result = getnameinfo (&remaddr.sa, addr_size,
  764.                                         remote_address,
  765.                                         sizeof (remote_address),
  766.                                         remote_service,
  767.                                         sizeof (remote_service),
  768.                                         NI_NUMERICHOST | NI_NUMERICSERV
  769.                                         | (tcp ? 0 : NI_DGRAM));
  770.                   if (result)
  771.                     continue;

  772.                   user_from_uid (user, sizeof (user), uid);

  773.                   buffer_xml_printf (
  774.                       buffer,
  775.                       "<item>"
  776.                       "<column name=\"local address\">%s</column>"
  777.                       "<column name=\"local port\">%s</column>"
  778.                       "<column name=\"remote address\">%s</column>"
  779.                       "<column name=\"remote port\">%s</column>"
  780.                       "<column name=\"state\">%s</column>"
  781.                       "<column name=\"user\">%s</column>"
  782.                       "<column name=\"family\">%s</column>"
  783.                       "<column name=\"protocol\">%s</column>"
  784.                       "</item>",
  785.                       local_address,
  786.                       local_service,
  787.                       remote_address,
  788.                       remote_service,
  789.                       format_socket_state (state),
  790.                       user,
  791.                       (family == AF_INET) ? "INET" : "INET6",
  792.                       tcp ? "STREAM" : "DGRAM");
  793.                 }
  794.             }
  795.         }
  796.       while (!feof (fp));

  797.       fclose (fp);
  798.     }
  799. }

  800. /* Collect data about internet sockets and write it into READBUF.  */

  801. static LONGEST
  802. linux_xfer_osdata_isockets (gdb_byte *readbuf,
  803.                             ULONGEST offset, ULONGEST len)
  804. {
  805.   static const char *buf;
  806.   static LONGEST len_avail = -1;
  807.   static struct buffer buffer;

  808.   if (offset == 0)
  809.     {
  810.       if (len_avail != -1 && len_avail != 0)
  811.         buffer_free (&buffer);
  812.       len_avail = 0;
  813.       buf = NULL;
  814.       buffer_init (&buffer);
  815.       buffer_grow_str (&buffer, "<osdata type=\"I sockets\">\n");

  816.       print_sockets (AF_INET, 1, &buffer);
  817.       print_sockets (AF_INET, 0, &buffer);
  818.       print_sockets (AF_INET6, 1, &buffer);
  819.       print_sockets (AF_INET6, 0, &buffer);

  820.       buffer_grow_str0 (&buffer, "</osdata>\n");
  821.       buf = buffer_finish (&buffer);
  822.       len_avail = strlen (buf);
  823.     }

  824.   if (offset >= len_avail)
  825.     {
  826.       /* Done.  Get rid of the buffer.  */
  827.       buffer_free (&buffer);
  828.       buf = NULL;
  829.       len_avail = 0;
  830.       return 0;
  831.     }

  832.   if (len > len_avail - offset)
  833.     len = len_avail - offset;
  834.   memcpy (readbuf, buf + offset, len);

  835.   return len;
  836. }

  837. /* Converts the time SECONDS into textual form and copies it into a
  838.    buffer TIME, with at most MAXLEN characters copied.  */

  839. static void
  840. time_from_time_t (char *time, int maxlen, TIME_T seconds)
  841. {
  842.   if (!seconds)
  843.     time[0] = '\0';
  844.   else
  845.     {
  846.       time_t t = (time_t) seconds;

  847.       strncpy (time, ctime (&t), maxlen);
  848.       time[maxlen - 1] = '\0';
  849.     }
  850. }

  851. /* Finds the group name for the group GID and copies it into GROUP.
  852.    At most MAXLEN characters are copied.  */

  853. static void
  854. group_from_gid (char *group, int maxlen, gid_t gid)
  855. {
  856.   struct group *grentry = getgrgid (gid);

  857.   if (grentry)
  858.     {
  859.       strncpy (group, grentry->gr_name, maxlen);
  860.       /* Ensure that the group name is null-terminated.  */
  861.       group[maxlen - 1] = '\0';
  862.     }
  863.   else
  864.     group[0] = '\0';
  865. }

  866. /* Collect data about shared memory recorded in /proc and write it
  867.    into READBUF.  */

  868. static LONGEST
  869. linux_xfer_osdata_shm (gdb_byte *readbuf,
  870.                        ULONGEST offset, ULONGEST len)
  871. {
  872.   static const char *buf;
  873.   static LONGEST len_avail = -1;
  874.   static struct buffer buffer;

  875.   if (offset == 0)
  876.     {
  877.       FILE *fp;

  878.       if (len_avail != -1 && len_avail != 0)
  879.         buffer_free (&buffer);
  880.       len_avail = 0;
  881.       buf = NULL;
  882.       buffer_init (&buffer);
  883.       buffer_grow_str (&buffer, "<osdata type=\"shared memory\">\n");

  884.       fp = gdb_fopen_cloexec ("/proc/sysvipc/shm", "r");
  885.       if (fp)
  886.         {
  887.           char buf[8192];

  888.           do
  889.             {
  890.               if (fgets (buf, sizeof (buf), fp))
  891.                 {
  892.                   key_t key;
  893.                   uid_t uid, cuid;
  894.                   gid_t gid, cgid;
  895.                   PID_T cpid, lpid;
  896.                   int shmid, size, nattch;
  897.                   TIME_T atime, dtime, ctime;
  898.                   unsigned int perms;
  899.                   int items_read;

  900.                   items_read = sscanf (buf,
  901.                                        "%d %d %o %d %lld %lld %d %u %u %u %u %lld %lld %lld",
  902.                                        &key, &shmid, &perms, &size,
  903.                                        &cpid, &lpid,
  904.                                        &nattch,
  905.                                        &uid, &gid, &cuid, &cgid,
  906.                                        &atime, &dtime, &ctime);

  907.                   if (items_read == 14)
  908.                     {
  909.                       char user[UT_NAMESIZE], group[UT_NAMESIZE];
  910.                       char cuser[UT_NAMESIZE], cgroup[UT_NAMESIZE];
  911.                       char ccmd[32], lcmd[32];
  912.                       char atime_str[32], dtime_str[32], ctime_str[32];

  913.                       user_from_uid (user, sizeof (user), uid);
  914.                       group_from_gid (group, sizeof (group), gid);
  915.                       user_from_uid (cuser, sizeof (cuser), cuid);
  916.                       group_from_gid (cgroup, sizeof (cgroup), cgid);

  917.                       command_from_pid (ccmd, sizeof (ccmd), cpid);
  918.                       command_from_pid (lcmd, sizeof (lcmd), lpid);

  919.                       time_from_time_t (atime_str, sizeof (atime_str), atime);
  920.                       time_from_time_t (dtime_str, sizeof (dtime_str), dtime);
  921.                       time_from_time_t (ctime_str, sizeof (ctime_str), ctime);

  922.                       buffer_xml_printf (
  923.                           &buffer,
  924.                           "<item>"
  925.                           "<column name=\"key\">%d</column>"
  926.                           "<column name=\"shmid\">%d</column>"
  927.                           "<column name=\"permissions\">%o</column>"
  928.                           "<column name=\"size\">%d</column>"
  929.                           "<column name=\"creator command\">%s</column>"
  930.                           "<column name=\"last op. command\">%s</column>"
  931.                           "<column name=\"num attached\">%d</column>"
  932.                           "<column name=\"user\">%s</column>"
  933.                           "<column name=\"group\">%s</column>"
  934.                           "<column name=\"creator user\">%s</column>"
  935.                           "<column name=\"creator group\">%s</column>"
  936.                           "<column name=\"last shmat() time\">%s</column>"
  937.                           "<column name=\"last shmdt() time\">%s</column>"
  938.                           "<column name=\"last shmctl() time\">%s</column>"
  939.                           "</item>",
  940.                           key,
  941.                           shmid,
  942.                           perms,
  943.                           size,
  944.                           ccmd,
  945.                           lcmd,
  946.                           nattch,
  947.                           user,
  948.                           group,
  949.                           cuser,
  950.                           cgroup,
  951.                           atime_str,
  952.                           dtime_str,
  953.                           ctime_str);
  954.                     }
  955.                 }
  956.             }
  957.           while (!feof (fp));

  958.           fclose (fp);
  959.         }

  960.       buffer_grow_str0 (&buffer, "</osdata>\n");
  961.       buf = buffer_finish (&buffer);
  962.       len_avail = strlen (buf);
  963.     }

  964.   if (offset >= len_avail)
  965.     {
  966.       /* Done.  Get rid of the buffer.  */
  967.       buffer_free (&buffer);
  968.       buf = NULL;
  969.       len_avail = 0;
  970.       return 0;
  971.     }

  972.   if (len > len_avail - offset)
  973.     len = len_avail - offset;
  974.   memcpy (readbuf, buf + offset, len);

  975.   return len;
  976. }

  977. /* Collect data about semaphores recorded in /proc and write it
  978.    into READBUF.  */

  979. static LONGEST
  980. linux_xfer_osdata_sem (gdb_byte *readbuf,
  981.                        ULONGEST offset, ULONGEST len)
  982. {
  983.   static const char *buf;
  984.   static LONGEST len_avail = -1;
  985.   static struct buffer buffer;

  986.   if (offset == 0)
  987.     {
  988.       FILE *fp;

  989.       if (len_avail != -1 && len_avail != 0)
  990.         buffer_free (&buffer);
  991.       len_avail = 0;
  992.       buf = NULL;
  993.       buffer_init (&buffer);
  994.       buffer_grow_str (&buffer, "<osdata type=\"semaphores\">\n");

  995.       fp = gdb_fopen_cloexec ("/proc/sysvipc/sem", "r");
  996.       if (fp)
  997.         {
  998.           char buf[8192];

  999.           do
  1000.             {
  1001.               if (fgets (buf, sizeof (buf), fp))
  1002.                 {
  1003.                   key_t key;
  1004.                   uid_t uid, cuid;
  1005.                   gid_t gid, cgid;
  1006.                   unsigned int perms, nsems;
  1007.                   int semid;
  1008.                   TIME_T otime, ctime;
  1009.                   int items_read;

  1010.                   items_read = sscanf (buf,
  1011.                                        "%d %d %o %u %d %d %d %d %lld %lld",
  1012.                                        &key, &semid, &perms, &nsems,
  1013.                                        &uid, &gid, &cuid, &cgid,
  1014.                                        &otime, &ctime);

  1015.                   if (items_read == 10)
  1016.                     {
  1017.                       char user[UT_NAMESIZE], group[UT_NAMESIZE];
  1018.                       char cuser[UT_NAMESIZE], cgroup[UT_NAMESIZE];
  1019.                       char otime_str[32], ctime_str[32];

  1020.                       user_from_uid (user, sizeof (user), uid);
  1021.                       group_from_gid (group, sizeof (group), gid);
  1022.                       user_from_uid (cuser, sizeof (cuser), cuid);
  1023.                       group_from_gid (cgroup, sizeof (cgroup), cgid);

  1024.                       time_from_time_t (otime_str, sizeof (otime_str), otime);
  1025.                       time_from_time_t (ctime_str, sizeof (ctime_str), ctime);

  1026.                       buffer_xml_printf (
  1027.                           &buffer,
  1028.                           "<item>"
  1029.                           "<column name=\"key\">%d</column>"
  1030.                           "<column name=\"semid\">%d</column>"
  1031.                           "<column name=\"permissions\">%o</column>"
  1032.                           "<column name=\"num semaphores\">%u</column>"
  1033.                           "<column name=\"user\">%s</column>"
  1034.                           "<column name=\"group\">%s</column>"
  1035.                           "<column name=\"creator user\">%s</column>"
  1036.                           "<column name=\"creator group\">%s</column>"
  1037.                           "<column name=\"last semop() time\">%s</column>"
  1038.                           "<column name=\"last semctl() time\">%s</column>"
  1039.                           "</item>",
  1040.                           key,
  1041.                           semid,
  1042.                           perms,
  1043.                           nsems,
  1044.                           user,
  1045.                           group,
  1046.                           cuser,
  1047.                           cgroup,
  1048.                           otime_str,
  1049.                           ctime_str);
  1050.                     }
  1051.                 }
  1052.             }
  1053.           while (!feof (fp));

  1054.           fclose (fp);
  1055.         }

  1056.       buffer_grow_str0 (&buffer, "</osdata>\n");
  1057.       buf = buffer_finish (&buffer);
  1058.       len_avail = strlen (buf);
  1059.     }

  1060.   if (offset >= len_avail)
  1061.     {
  1062.       /* Done.  Get rid of the buffer.  */
  1063.       buffer_free (&buffer);
  1064.       buf = NULL;
  1065.       len_avail = 0;
  1066.       return 0;
  1067.     }

  1068.   if (len > len_avail - offset)
  1069.     len = len_avail - offset;
  1070.   memcpy (readbuf, buf + offset, len);

  1071.   return len;
  1072. }

  1073. /* Collect data about message queues recorded in /proc and write it
  1074.    into READBUF.  */

  1075. static LONGEST
  1076. linux_xfer_osdata_msg (gdb_byte *readbuf,
  1077.                        ULONGEST offset, ULONGEST len)
  1078. {
  1079.   static const char *buf;
  1080.   static LONGEST len_avail = -1;
  1081.   static struct buffer buffer;

  1082.   if (offset == 0)
  1083.     {
  1084.       FILE *fp;

  1085.       if (len_avail != -1 && len_avail != 0)
  1086.         buffer_free (&buffer);
  1087.       len_avail = 0;
  1088.       buf = NULL;
  1089.       buffer_init (&buffer);
  1090.       buffer_grow_str (&buffer, "<osdata type=\"message queues\">\n");

  1091.       fp = gdb_fopen_cloexec ("/proc/sysvipc/msg", "r");
  1092.       if (fp)
  1093.         {
  1094.           char buf[8192];

  1095.           do
  1096.             {
  1097.               if (fgets (buf, sizeof (buf), fp))
  1098.                 {
  1099.                   key_t key;
  1100.                   PID_T lspid, lrpid;
  1101.                   uid_t uid, cuid;
  1102.                   gid_t gid, cgid;
  1103.                   unsigned int perms, cbytes, qnum;
  1104.                   int msqid;
  1105.                   TIME_T stime, rtime, ctime;
  1106.                   int items_read;

  1107.                   items_read = sscanf (buf,
  1108.                                        "%d %d %o %u %u %lld %lld %d %d %d %d %lld %lld %lld",
  1109.                                        &key, &msqid, &perms, &cbytes, &qnum,
  1110.                                        &lspid, &lrpid, &uid, &gid, &cuid, &cgid,
  1111.                                        &stime, &rtime, &ctime);

  1112.                   if (items_read == 14)
  1113.                     {
  1114.                       char user[UT_NAMESIZE], group[UT_NAMESIZE];
  1115.                       char cuser[UT_NAMESIZE], cgroup[UT_NAMESIZE];
  1116.                       char lscmd[32], lrcmd[32];
  1117.                       char stime_str[32], rtime_str[32], ctime_str[32];

  1118.                       user_from_uid (user, sizeof (user), uid);
  1119.                       group_from_gid (group, sizeof (group), gid);
  1120.                       user_from_uid (cuser, sizeof (cuser), cuid);
  1121.                       group_from_gid (cgroup, sizeof (cgroup), cgid);

  1122.                       command_from_pid (lscmd, sizeof (lscmd), lspid);
  1123.                       command_from_pid (lrcmd, sizeof (lrcmd), lrpid);

  1124.                       time_from_time_t (stime_str, sizeof (stime_str), stime);
  1125.                       time_from_time_t (rtime_str, sizeof (rtime_str), rtime);
  1126.                       time_from_time_t (ctime_str, sizeof (ctime_str), ctime);

  1127.                       buffer_xml_printf (
  1128.                           &buffer,
  1129.                           "<item>"
  1130.                           "<column name=\"key\">%d</column>"
  1131.                           "<column name=\"msqid\">%d</column>"
  1132.                           "<column name=\"permissions\">%o</column>"
  1133.                           "<column name=\"num used bytes\">%u</column>"
  1134.                           "<column name=\"num messages\">%u</column>"
  1135.                           "<column name=\"last msgsnd() command\">%s</column>"
  1136.                           "<column name=\"last msgrcv() command\">%s</column>"
  1137.                           "<column name=\"user\">%s</column>"
  1138.                           "<column name=\"group\">%s</column>"
  1139.                           "<column name=\"creator user\">%s</column>"
  1140.                           "<column name=\"creator group\">%s</column>"
  1141.                           "<column name=\"last msgsnd() time\">%s</column>"
  1142.                           "<column name=\"last msgrcv() time\">%s</column>"
  1143.                           "<column name=\"last msgctl() time\">%s</column>"
  1144.                           "</item>",
  1145.                           key,
  1146.                           msqid,
  1147.                           perms,
  1148.                           cbytes,
  1149.                           qnum,
  1150.                           lscmd,
  1151.                           lrcmd,
  1152.                           user,
  1153.                           group,
  1154.                           cuser,
  1155.                           cgroup,
  1156.                           stime_str,
  1157.                           rtime_str,
  1158.                           ctime_str);
  1159.                     }
  1160.                 }
  1161.             }
  1162.           while (!feof (fp));

  1163.           fclose (fp);
  1164.         }

  1165.       buffer_grow_str0 (&buffer, "</osdata>\n");
  1166.       buf = buffer_finish (&buffer);
  1167.       len_avail = strlen (buf);
  1168.     }

  1169.   if (offset >= len_avail)
  1170.     {
  1171.       /* Done.  Get rid of the buffer.  */
  1172.       buffer_free (&buffer);
  1173.       buf = NULL;
  1174.       len_avail = 0;
  1175.       return 0;
  1176.     }

  1177.   if (len > len_avail - offset)
  1178.     len = len_avail - offset;
  1179.   memcpy (readbuf, buf + offset, len);

  1180.   return len;
  1181. }

  1182. /* Collect data about loaded kernel modules and write it into
  1183.    READBUF.  */

  1184. static LONGEST
  1185. linux_xfer_osdata_modules (gdb_byte *readbuf,
  1186.                            ULONGEST offset, ULONGEST len)
  1187. {
  1188.   static const char *buf;
  1189.   static LONGEST len_avail = -1;
  1190.   static struct buffer buffer;

  1191.   if (offset == 0)
  1192.     {
  1193.       FILE *fp;

  1194.       if (len_avail != -1 && len_avail != 0)
  1195.         buffer_free (&buffer);
  1196.       len_avail = 0;
  1197.       buf = NULL;
  1198.       buffer_init (&buffer);
  1199.       buffer_grow_str (&buffer, "<osdata type=\"modules\">\n");

  1200.       fp = gdb_fopen_cloexec ("/proc/modules", "r");
  1201.       if (fp)
  1202.         {
  1203.           char buf[8192];

  1204.           do
  1205.             {
  1206.               if (fgets (buf, sizeof (buf), fp))
  1207.                 {
  1208.                   char *name, *dependencies, *status, *tmp;
  1209.                   unsigned int size;
  1210.                   unsigned long long address;
  1211.                   int uses;

  1212.                   name = strtok (buf, " ");
  1213.                   if (name == NULL)
  1214.                     continue;

  1215.                   tmp = strtok (NULL, " ");
  1216.                   if (tmp == NULL)
  1217.                     continue;
  1218.                   if (sscanf (tmp, "%u", &size) != 1)
  1219.                     continue;

  1220.                   tmp = strtok (NULL, " ");
  1221.                   if (tmp == NULL)
  1222.                     continue;
  1223.                   if (sscanf (tmp, "%d", &uses) != 1)
  1224.                     continue;

  1225.                   dependencies = strtok (NULL, " ");
  1226.                   if (dependencies == NULL)
  1227.                     continue;

  1228.                   status = strtok (NULL, " ");
  1229.                   if (status == NULL)
  1230.                     continue;

  1231.                   tmp = strtok (NULL, "\n");
  1232.                   if (tmp == NULL)
  1233.                     continue;
  1234.                   if (sscanf (tmp, "%llx", &address) != 1)
  1235.                     continue;

  1236.                   buffer_xml_printf (
  1237.                         &buffer,
  1238.                         "<item>"
  1239.                         "<column name=\"name\">%s</column>"
  1240.                         "<column name=\"size\">%u</column>"
  1241.                         "<column name=\"num uses\">%d</column>"
  1242.                         "<column name=\"dependencies\">%s</column>"
  1243.                         "<column name=\"status\">%s</column>"
  1244.                         "<column name=\"address\">%llx</column>"
  1245.                         "</item>",
  1246.                         name,
  1247.                         size,
  1248.                         uses,
  1249.                         dependencies,
  1250.                         status,
  1251.                         address);
  1252.                 }
  1253.             }
  1254.           while (!feof (fp));

  1255.           fclose (fp);
  1256.         }

  1257.       buffer_grow_str0 (&buffer, "</osdata>\n");
  1258.       buf = buffer_finish (&buffer);
  1259.       len_avail = strlen (buf);
  1260.     }

  1261.   if (offset >= len_avail)
  1262.     {
  1263.       /* Done.  Get rid of the buffer.  */
  1264.       buffer_free (&buffer);
  1265.       buf = NULL;
  1266.       len_avail = 0;
  1267.       return 0;
  1268.     }

  1269.   if (len > len_avail - offset)
  1270.     len = len_avail - offset;
  1271.   memcpy (readbuf, buf + offset, len);

  1272.   return len;
  1273. }

  1274. struct osdata_type {
  1275.   char *type;
  1276.   char *title;
  1277.   char *description;
  1278.   LONGEST (*getter) (gdb_byte *readbuf, ULONGEST offset, ULONGEST len);
  1279. } osdata_table[] = {
  1280.   { "processes", "Processes", "Listing of all processes",
  1281.     linux_xfer_osdata_processes },
  1282.   { "procgroups", "Process groups", "Listing of all process groups",
  1283.     linux_xfer_osdata_processgroups },
  1284.   { "threads", "Threads", "Listing of all threads",
  1285.     linux_xfer_osdata_threads },
  1286.   { "files", "File descriptors", "Listing of all file descriptors",
  1287.     linux_xfer_osdata_fds },
  1288.   { "sockets", "Sockets", "Listing of all internet-domain sockets",
  1289.     linux_xfer_osdata_isockets },
  1290.   { "shm", "Shared-memory regions", "Listing of all shared-memory regions",
  1291.     linux_xfer_osdata_shm },
  1292.   { "semaphores", "Semaphores", "Listing of all semaphores",
  1293.     linux_xfer_osdata_sem },
  1294.   { "msg", "Message queues", "Listing of all message queues",
  1295.     linux_xfer_osdata_msg },
  1296.   { "modules", "Kernel modules", "Listing of all loaded kernel modules",
  1297.     linux_xfer_osdata_modules },
  1298.   { NULL, NULL, NULL }
  1299. };

  1300. LONGEST
  1301. linux_common_xfer_osdata (const char *annex, gdb_byte *readbuf,
  1302.                           ULONGEST offset, ULONGEST len)
  1303. {
  1304.   if (!annex || *annex == '\0')
  1305.     {
  1306.       static const char *buf;
  1307.       static LONGEST len_avail = -1;
  1308.       static struct buffer buffer;

  1309.       if (offset == 0)
  1310.         {
  1311.           int i;

  1312.           if (len_avail != -1 && len_avail != 0)
  1313.             buffer_free (&buffer);
  1314.           len_avail = 0;
  1315.           buf = NULL;
  1316.           buffer_init (&buffer);
  1317.           buffer_grow_str (&buffer, "<osdata type=\"types\">\n");

  1318.           for (i = 0; osdata_table[i].type; ++i)
  1319.             buffer_xml_printf (
  1320.                                &buffer,
  1321.                                "<item>"
  1322.                                "<column name=\"Type\">%s</column>"
  1323.                                "<column name=\"Description\">%s</column>"
  1324.                                "<column name=\"Title\">%s</column>"
  1325.                                "</item>",
  1326.                                osdata_table[i].type,
  1327.                                osdata_table[i].description,
  1328.                                osdata_table[i].title);

  1329.           buffer_grow_str0 (&buffer, "</osdata>\n");
  1330.           buf = buffer_finish (&buffer);
  1331.           len_avail = strlen (buf);
  1332.         }

  1333.       if (offset >= len_avail)
  1334.         {
  1335.           /* Done.  Get rid of the buffer.  */
  1336.           buffer_free (&buffer);
  1337.           buf = NULL;
  1338.           len_avail = 0;
  1339.           return 0;
  1340.         }

  1341.       if (len > len_avail - offset)
  1342.         len = len_avail - offset;
  1343.       memcpy (readbuf, buf + offset, len);

  1344.       return len;
  1345.     }
  1346.   else
  1347.     {
  1348.       int i;

  1349.       for (i = 0; osdata_table[i].type; ++i)
  1350.         {
  1351.           if (strcmp (annex, osdata_table[i].type) == 0)
  1352.             {
  1353.               gdb_assert (readbuf);

  1354.               return (osdata_table[i].getter) (readbuf, offset, len);
  1355.             }
  1356.         }

  1357.       return 0;
  1358.     }
  1359. }