446cf2
commit 647103ea3a2aae2e6791b3bebe9e33eedc168102
446cf2
Author: Florian Weimer <fweimer@redhat.com>
446cf2
Date:   Fri Oct 9 10:13:14 2020 +0200
446cf2
446cf2
    elf: Enhance ld.so --help to print HWCAP subdirectories
446cf2
    
446cf2
    Reviewed-by: Adhemerval Zanella  <adhemerval.zanella@linaro.org>
446cf2
446cf2
diff --git a/elf/dl-usage.c b/elf/dl-usage.c
446cf2
index 8d39bc9d5442bc59..c07f43835bd771cf 100644
446cf2
--- a/elf/dl-usage.c
446cf2
+++ b/elf/dl-usage.c
446cf2
@@ -22,6 +22,8 @@
446cf2
 #include <unistd.h>
446cf2
 #include "version.h"
446cf2
 
446cf2
+#include <dl-hwcaps.h>
446cf2
+
446cf2
 void
446cf2
 _dl_usage (const char *argv0, const char *wrong_option)
446cf2
 {
446cf2
@@ -101,6 +103,65 @@ print_search_path_for_help (struct dl_main_state *state)
446cf2
   print_search_path_for_help_1 (__rtld_search_dirs.dirs);
446cf2
 }
446cf2
 
446cf2
+/* Helper function for printing flags associated with a HWCAP name.  */
446cf2
+static void
446cf2
+print_hwcap_1 (bool *first, bool active, const char *label)
446cf2
+{
446cf2
+  if (active)
446cf2
+    {
446cf2
+      if (*first)
446cf2
+        {
446cf2
+          _dl_printf (" (");
446cf2
+          *first = false;
446cf2
+        }
446cf2
+      else
446cf2
+        _dl_printf (", ");
446cf2
+      _dl_printf ("%s", label);
446cf2
+    }
446cf2
+}
446cf2
+
446cf2
+/* Called after a series of print_hwcap_1 calls to emit the line
446cf2
+   terminator.  */
446cf2
+static void
446cf2
+print_hwcap_1_finish (bool *first)
446cf2
+{
446cf2
+  if (*first)
446cf2
+    _dl_printf ("\n");
446cf2
+  else
446cf2
+    _dl_printf (")\n");
446cf2
+}
446cf2
+
446cf2
+/* Write a list of hwcap subdirectories to standard output.  See
446cf2
+ _dl_important_hwcaps in dl-hwcaps.c.  */
446cf2
+static void
446cf2
+print_legacy_hwcap_directories (void)
446cf2
+{
446cf2
+  _dl_printf ("\n\
446cf2
+Legacy HWCAP subdirectories under library search path directories:\n");
446cf2
+
446cf2
+  const char *platform = GLRO (dl_platform);
446cf2
+  if (platform != NULL)
446cf2
+    _dl_printf ("  %s (AT_PLATFORM; supported, searched)\n", platform);
446cf2
+
446cf2
+  _dl_printf ("  tls (supported, searched)\n");
446cf2
+
446cf2
+  uint64_t hwcap_mask = GET_HWCAP_MASK();
446cf2
+  uint64_t searched = GLRO (dl_hwcap) & hwcap_mask;
446cf2
+  for (int n = 63; n >= 0; --n)
446cf2
+    {
446cf2
+      uint64_t bit = 1ULL << n;
446cf2
+      if (HWCAP_IMPORTANT & bit)
446cf2
+        {
446cf2
+          _dl_printf ("  %s", _dl_hwcap_string (n));
446cf2
+          bool first = true;
446cf2
+          print_hwcap_1 (&first, GLRO (dl_hwcap) & bit, "supported");
446cf2
+          print_hwcap_1 (&first, !(hwcap_mask & bit), "masked");
446cf2
+          print_hwcap_1 (&first, searched & bit, "searched");
446cf2
+          print_hwcap_1_finish (&first);
446cf2
+        }
446cf2
+    }
446cf2
+}
446cf2
+
446cf2
 void
446cf2
 _dl_help (const char *argv0, struct dl_main_state *state)
446cf2
 {
446cf2
@@ -136,5 +197,6 @@ This program interpreter self-identifies as: " RTLD "\n\
446cf2
 ",
446cf2
               argv0);
446cf2
   print_search_path_for_help (state);
446cf2
+  print_legacy_hwcap_directories ();
446cf2
   _exit (EXIT_SUCCESS);
446cf2
 }