gdb/mi: include ranges in =library-unloaded event

Having looked at the dlmopen support in GDB, it occurred to me that
the current MI =library-unloaded event doesn't incude enough
information to be useful.

Consider the gdb.mi/mi-dlmopen.exp test, this test loads libraries
into multiple linker namespaces, and then unloads these libraries.

We should probably figure out a way to include the linker namepsace ID
in GDB's output, e.g. in the =library-loaded and =library-unloaded MI
events, and in the output of 'info sharedlibrary'.  But this commit is
not about doing that.

This commit includes the 'ranges' information in the =library-unloaded
event output.  This is the same ranges information as is included in
the =library-loaded output.  Even without the linker namespace ID,
this should allow MI consumers to figure out which library instance is
being unloaded.

Here is the 'info sharedlibrary' output for mi-dlmopen.exp at the
point where all the shared libraries are loaded:

  info sharedlibrary
  &"info sharedlibrary\n"
  ~"From                To                  Syms Read   Shared Object Library\n"
  ~"0x00007ffff7fca000  0x00007ffff7ff03f5  Yes         /lib64/ld-linux-x86-64.so.2\n"
  ~"0x00007ffff7eda3d0  0x00007ffff7f4e898  Yes         /lib64/libm.so.6\n"
  ~"0x00007ffff7d0e800  0x00007ffff7e6dccd  Yes         /lib64/libc.so.6\n"
  ~"0x00007ffff7fbd040  0x00007ffff7fbd116  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so\n"
  ~"0x00007ffff7fb8040  0x00007ffff7fb80f9  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib-dep.so\n"
  ~"0x00007ffff7bfe3d0  0x00007ffff7c72898  Yes         /lib64/libm.so.6\n"
  ~"0x00007ffff7a32800  0x00007ffff7b91ccd  Yes         /lib64/libc.so.6\n"
  ~"0x00007ffff7fca000  0x00007ffff7ff03f5  Yes         /lib64/ld-linux-x86-64.so.2\n"
  ~"0x00007ffff7fb3040  0x00007ffff7fb3116  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so\n"
  ~"0x00007ffff7fae040  0x00007ffff7fae0f9  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib-dep.so\n"
  ~"0x00007ffff7ce1040  0x00007ffff7ce1116  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so\n"
  ~"0x00007ffff7cdc040  0x00007ffff7cdc0f9  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib-dep.so\n"
  ~"0x00007ffff79253d0  0x00007ffff7999898  Yes         /lib64/libm.so.6\n"
  ~"0x00007ffff7759800  0x00007ffff78b8ccd  Yes         /lib64/libc.so.6\n"
  ~"0x00007ffff7fca000  0x00007ffff7ff03f5  Yes         /lib64/ld-linux-x86-64.so.2\n"
  ~"0x00007ffff7cd7040  0x00007ffff7cd7116  Yes         /tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.2.so\n"
  ^done
  (gdb)

Notice that dlmopen-lib.1.so is loaded multiple times.  Here is the
=library-unloaded event when one copy of this library is unloaded
before this patch:

  =library-unloaded,id="/tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so",
                    target-name="/tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so",
		    host-name="/tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so",
		    thread-group="i1",

It is not possible, given this information, to know which copy of
dlmopen-lib.1.so has actually been unloaded.  An MI consumer would
need to query the full shared library list and update from that
information.

After this patch the new output is:

  =library-unloaded,id="/tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so",
                    target-name="/tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so",
		    host-name="/tmp/build/gdb/testsuite/outputs/gdb.mi/mi-dlmopen/dlmopen-lib.1.so",
		    thread-group="i1",
		    ranges=[{from="0x00007ffff7fbd040",to="0x00007ffff7fbd116"}],
		    still-in-use="false"

The new 'ranges' field allows an MI consumer to uniquely identify
which library instance was just unmapped.  A frontent could,
e.g. update a library list with no need to query the full shared
library list.

To include the 'ranges' field I updated mi_interp::on_solib_unloaded
to call a new helper function.  The new helper function is split out
from the existing mi_output_solib_attribs.  I was tempted to just call
mi_output_solib_attribs, but doing so would mean that the
'symbols-loaded' field was also added to the =library-unloaded event,
however, the docs for 'symbols-unloaded' on =library-loaded says:

  The @var{symbols-loaded} field is emitted only for backward
  compatibility and should not be relied on to convey any useful
  information.

And it seemed silly to add a fields to =library-unloaded, which I
would then document as something that should be ignored.  The new
helper function means I can avoid emitting the 'symbols-loaded'
field.

I have also added a 'still-in-use' field.  When true this indicates
that the library was removed from the inferior's library list, but the
mapping was not removed from the inferior's address space as there is
another copy of the library that is still using the library.  In the
above list, notice that ld-linux-x86-64.so.2 appears 3 times, but each
instance is mapped as 0x00007ffff7fca000.  When one copy of
ld-linux-x86-64.so.2 is unloaded, here's the event:

  =library-unloaded,id="/lib64/ld-linux-x86-64.so.2",
                    target-name="/lib64/ld-linux-x86-64.so.2",
		    host-name="/lib64/ld-linux-x86-64.so.2",
		    thread-group="i1",
		    ranges=[{from="0x00007ffff7fca000",to="0x00007ffff7ff03f5"}],
		    still-in-use="true"

The 'still-in-use' field is 'true', this indicates there are at least
one instance of this library remaining mapped at 0x00007ffff7fca000.

Reviewed-By: Eli Zaretskii <eliz@gnu.org>
4 files changed