b1dca6
commit 8f7a75d700af809eeb4363895078fabfb3a9d7c3
b1dca6
Author: Florian Weimer <fweimer@redhat.com>
b1dca6
Date:   Mon Feb 17 16:49:40 2020 +0100
b1dca6
b1dca6
    elf: Implement DT_AUDIT, DT_DEPAUDIT support [BZ #24943]
b1dca6
    
b1dca6
    binutils ld has supported --audit, --depaudit for a long time,
b1dca6
    only support in glibc has been missing.
b1dca6
    
b1dca6
    Reviewed-by: Carlos O'Donell <carlos@redhat.com>
b1dca6
b1dca6
Conflicts:
b1dca6
	elf/Makefile
b1dca6
	  (Test backport differences.)
b1dca6
b1dca6
diff --git a/elf/Makefile b/elf/Makefile
b1dca6
index 5b0aeccb0a53c182..a6601ba84c8f4017 100644
b1dca6
--- a/elf/Makefile
b1dca6
+++ b/elf/Makefile
b1dca6
@@ -195,7 +195,8 @@ tests += restest1 preloadtest loadfail multiload origtest resolvfail \
b1dca6
 	 tst-sonamemove-link tst-sonamemove-dlopen \
b1dca6
 	 tst-auditmany tst-initfinilazyfail \
b1dca6
 	 tst-dlopenfail tst-dlopenfail-2 \
b1dca6
-	 tst-filterobj tst-filterobj-dlopen tst-auxobj tst-auxobj-dlopen
b1dca6
+	 tst-filterobj tst-filterobj-dlopen tst-auxobj tst-auxobj-dlopen \
b1dca6
+	 tst-audit14 tst-audit15 tst-audit16
b1dca6
 #	 reldep9
b1dca6
 tests-internal += loadtest unload unload2 circleload1 \
b1dca6
 	 neededtest neededtest2 neededtest3 neededtest4 \
b1dca6
@@ -310,7 +311,8 @@ modules-names = testobj1 testobj2 testobj3 testobj4 testobj5 testobj6 \
b1dca6
 		tst-initlazyfailmod tst-finilazyfailmod \
b1dca6
 		tst-dlopenfailmod1 tst-dlopenfaillinkmod tst-dlopenfailmod2 \
b1dca6
 		tst-dlopenfailmod3 \
b1dca6
-		tst-filterobj-flt tst-filterobj-aux tst-filterobj-filtee
b1dca6
+		tst-filterobj-flt tst-filterobj-aux tst-filterobj-filtee \
b1dca6
+		tst-auditlogmod-1 tst-auditlogmod-2 tst-auditlogmod-3
b1dca6
 
b1dca6
 # Most modules build with _ISOMAC defined, but those filtered out
b1dca6
 # depend on internal headers.
b1dca6
@@ -1453,6 +1455,22 @@ $(objpfx)tst-auditmany.out: $(objpfx)tst-auditmanymod1.so \
b1dca6
 tst-auditmany-ENV = \
b1dca6
   LD_AUDIT=tst-auditmanymod1.so:tst-auditmanymod2.so:tst-auditmanymod3.so:tst-auditmanymod4.so:tst-auditmanymod5.so:tst-auditmanymod6.so:tst-auditmanymod7.so:tst-auditmanymod8.so:tst-auditmanymod9.so
b1dca6
 
b1dca6
+LDFLAGS-tst-audit14 = -Wl,--audit=tst-auditlogmod-1.so
b1dca6
+$(objpfx)tst-auditlogmod-1.so: $(libsupport)
b1dca6
+$(objpfx)tst-audit14.out: $(objpfx)tst-auditlogmod-1.so
b1dca6
+LDFLAGS-tst-audit15 = \
b1dca6
+  -Wl,--audit=tst-auditlogmod-1.so,--depaudit=tst-auditlogmod-2.so
b1dca6
+$(objpfx)tst-auditlogmod-2.so: $(libsupport)
b1dca6
+$(objpfx)tst-audit15.out: \
b1dca6
+  $(objpfx)tst-auditlogmod-1.so $(objpfx)tst-auditlogmod-2.so
b1dca6
+LDFLAGS-tst-audit16 = \
b1dca6
+  -Wl,--audit=tst-auditlogmod-1.so:tst-auditlogmod-2.so \
b1dca6
+  -Wl,--depaudit=tst-auditlogmod-3.so
b1dca6
+$(objpfx)tst-auditlogmod-3.so: $(libsupport)
b1dca6
+$(objpfx)tst-audit16.out: \
b1dca6
+  $(objpfx)tst-auditlogmod-1.so $(objpfx)tst-auditlogmod-2.so \
b1dca6
+  $(objpfx)tst-auditlogmod-3.so
b1dca6
+
b1dca6
 # tst-sonamemove links against an older implementation of the library.
b1dca6
 LDFLAGS-tst-sonamemove-linkmod1.so = \
b1dca6
   -Wl,--version-script=tst-sonamemove-linkmod1.map \
b1dca6
diff --git a/elf/rtld.c b/elf/rtld.c
b1dca6
index c39cb8f2cd4bb1cc..d44facf5343b3301 100644
b1dca6
--- a/elf/rtld.c
b1dca6
+++ b/elf/rtld.c
b1dca6
@@ -151,9 +151,17 @@ static void audit_list_init (struct audit_list *);
b1dca6
    not be called after audit_list_next.  */
b1dca6
 static void audit_list_add_string (struct audit_list *, const char *);
b1dca6
 
b1dca6
+/* Add the audit strings from the link map, found in the dynamic
b1dca6
+   segment at TG (either DT_AUDIT and DT_DEPAUDIT).  Must be called
b1dca6
+   before audit_list_next.  */
b1dca6
+static void audit_list_add_dynamic_tag (struct audit_list *,
b1dca6
+					struct link_map *,
b1dca6
+					unsigned int tag);
b1dca6
+
b1dca6
 /* Extract the next audit module from the audit list.  Only modules
b1dca6
    for which dso_name_valid_for_suid is true are returned.  Must be
b1dca6
-   called after all the audit_list_add_string calls.  */
b1dca6
+   called after all the audit_list_add_string,
b1dca6
+   audit_list_add_dynamic_tags calls.  */
b1dca6
 static const char *audit_list_next (struct audit_list *);
b1dca6
 
b1dca6
 /* This is a list of all the modes the dynamic loader can be in.  */
b1dca6
@@ -233,6 +241,16 @@ audit_list_add_string (struct audit_list *list, const char *string)
b1dca6
     list->current_tail = string;
b1dca6
 }
b1dca6
 
b1dca6
+static void
b1dca6
+audit_list_add_dynamic_tag (struct audit_list *list, struct link_map *main_map,
b1dca6
+			    unsigned int tag)
b1dca6
+{
b1dca6
+  ElfW(Dyn) *info = main_map->l_info[ADDRIDX (tag)];
b1dca6
+  const char *strtab = (const char *) D_PTR (main_map, l_info[DT_STRTAB]);
b1dca6
+  if (info != NULL)
b1dca6
+    audit_list_add_string (list, strtab + info->d_un.d_val);
b1dca6
+}
b1dca6
+
b1dca6
 static const char *
b1dca6
 audit_list_next (struct audit_list *list)
b1dca6
 {
b1dca6
@@ -1630,6 +1648,9 @@ ERROR: '%s': cannot process note segment.\n", _dl_argv[0]);
b1dca6
     /* Assign a module ID.  Do this before loading any audit modules.  */
b1dca6
     GL(dl_rtld_map).l_tls_modid = _dl_next_tls_modid ();
b1dca6
 
b1dca6
+  audit_list_add_dynamic_tag (&audit_list, main_map, DT_AUDIT);
b1dca6
+  audit_list_add_dynamic_tag (&audit_list, main_map, DT_DEPAUDIT);
b1dca6
+
b1dca6
   /* If we have auditing DSOs to load, do it now.  */
b1dca6
   bool need_security_init = true;
b1dca6
   if (audit_list.length > 0)
b1dca6
diff --git a/elf/tst-audit14.c b/elf/tst-audit14.c
b1dca6
new file mode 100644
b1dca6
index 0000000000000000..27ff8db9480a98cc
b1dca6
--- /dev/null
b1dca6
+++ b/elf/tst-audit14.c
b1dca6
@@ -0,0 +1,46 @@
b1dca6
+/* Main program with DT_AUDIT.  One audit module.
b1dca6
+   Copyright (C) 2020 Free Software Foundation, Inc.
b1dca6
+   This file is part of the GNU C Library.
b1dca6
+
b1dca6
+   The GNU C Library is free software; you can redistribute it and/or
b1dca6
+   modify it under the terms of the GNU Lesser General Public
b1dca6
+   License as published by the Free Software Foundation; either
b1dca6
+   version 2.1 of the License, or (at your option) any later version.
b1dca6
+
b1dca6
+   The GNU C Library is distributed in the hope that it will be useful,
b1dca6
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
b1dca6
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
b1dca6
+   Lesser General Public License for more details.
b1dca6
+
b1dca6
+   You should have received a copy of the GNU Lesser General Public
b1dca6
+   License along with the GNU C Library; if not, see
b1dca6
+   <https://www.gnu.org/licenses/>.  */
b1dca6
+
b1dca6
+#include <stdlib.h>
b1dca6
+#include <string.h>
b1dca6
+#include <support/check.h>
b1dca6
+#include <support/xstdio.h>
b1dca6
+
b1dca6
+static int
b1dca6
+do_test (void)
b1dca6
+{
b1dca6
+  /* Verify what the audit module has written.  This test assumes that
b1dca6
+     standard output has been redirected to a regular file.  */
b1dca6
+  FILE *fp = xfopen ("/dev/stdout", "r");
b1dca6
+
b1dca6
+  char *buffer = NULL;
b1dca6
+  size_t buffer_length = 0;
b1dca6
+  size_t line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  const char *message = "info: tst-auditlogmod-1.so loaded\n";
b1dca6
+  TEST_COMPARE_BLOB (message, strlen (message), buffer, line_length);
b1dca6
+
b1dca6
+  /* No more audit module output.  */
b1dca6
+  line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  TEST_COMPARE_BLOB ("", 0, buffer, line_length);
b1dca6
+
b1dca6
+  free (buffer);
b1dca6
+  xfclose (fp);
b1dca6
+  return 0;
b1dca6
+}
b1dca6
+
b1dca6
+#include <support/test-driver.c>
b1dca6
diff --git a/elf/tst-audit15.c b/elf/tst-audit15.c
b1dca6
new file mode 100644
b1dca6
index 0000000000000000..cbd1589ec40653d1
b1dca6
--- /dev/null
b1dca6
+++ b/elf/tst-audit15.c
b1dca6
@@ -0,0 +1,50 @@
b1dca6
+/* Main program with DT_AUDIT and DT_DEPAUDIT.  Two audit modules.
b1dca6
+   Copyright (C) 2020 Free Software Foundation, Inc.
b1dca6
+   This file is part of the GNU C Library.
b1dca6
+
b1dca6
+   The GNU C Library is free software; you can redistribute it and/or
b1dca6
+   modify it under the terms of the GNU Lesser General Public
b1dca6
+   License as published by the Free Software Foundation; either
b1dca6
+   version 2.1 of the License, or (at your option) any later version.
b1dca6
+
b1dca6
+   The GNU C Library is distributed in the hope that it will be useful,
b1dca6
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
b1dca6
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
b1dca6
+   Lesser General Public License for more details.
b1dca6
+
b1dca6
+   You should have received a copy of the GNU Lesser General Public
b1dca6
+   License along with the GNU C Library; if not, see
b1dca6
+   <https://www.gnu.org/licenses/>.  */
b1dca6
+
b1dca6
+#include <stdlib.h>
b1dca6
+#include <string.h>
b1dca6
+#include <support/check.h>
b1dca6
+#include <support/xstdio.h>
b1dca6
+
b1dca6
+static int
b1dca6
+do_test (void)
b1dca6
+{
b1dca6
+  /* Verify what the audit modules have written.  This test assumes
b1dca6
+     that standard output has been redirected to a regular file.  */
b1dca6
+  FILE *fp = xfopen ("/dev/stdout", "r");
b1dca6
+
b1dca6
+  char *buffer = NULL;
b1dca6
+  size_t buffer_length = 0;
b1dca6
+  size_t line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  const char *message = "info: tst-auditlogmod-1.so loaded\n";
b1dca6
+  TEST_COMPARE_BLOB (message, strlen (message), buffer, line_length);
b1dca6
+
b1dca6
+  line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  message = "info: tst-auditlogmod-2.so loaded\n";
b1dca6
+  TEST_COMPARE_BLOB (message, strlen (message), buffer, line_length);
b1dca6
+
b1dca6
+  /* No more audit module output.  */
b1dca6
+  line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  TEST_COMPARE_BLOB ("", 0, buffer, line_length);
b1dca6
+
b1dca6
+  free (buffer);
b1dca6
+  xfclose (fp);
b1dca6
+  return 0;
b1dca6
+}
b1dca6
+
b1dca6
+#include <support/test-driver.c>
b1dca6
diff --git a/elf/tst-audit16.c b/elf/tst-audit16.c
b1dca6
new file mode 100644
b1dca6
index 0000000000000000..dd6ce189ea6fffa3
b1dca6
--- /dev/null
b1dca6
+++ b/elf/tst-audit16.c
b1dca6
@@ -0,0 +1,54 @@
b1dca6
+/* Main program with DT_AUDIT and DT_DEPAUDIT.  Three audit modules.
b1dca6
+   Copyright (C) 2020 Free Software Foundation, Inc.
b1dca6
+   This file is part of the GNU C Library.
b1dca6
+
b1dca6
+   The GNU C Library is free software; you can redistribute it and/or
b1dca6
+   modify it under the terms of the GNU Lesser General Public
b1dca6
+   License as published by the Free Software Foundation; either
b1dca6
+   version 2.1 of the License, or (at your option) any later version.
b1dca6
+
b1dca6
+   The GNU C Library is distributed in the hope that it will be useful,
b1dca6
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
b1dca6
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
b1dca6
+   Lesser General Public License for more details.
b1dca6
+
b1dca6
+   You should have received a copy of the GNU Lesser General Public
b1dca6
+   License along with the GNU C Library; if not, see
b1dca6
+   <https://www.gnu.org/licenses/>.  */
b1dca6
+
b1dca6
+#include <stdlib.h>
b1dca6
+#include <string.h>
b1dca6
+#include <support/check.h>
b1dca6
+#include <support/xstdio.h>
b1dca6
+
b1dca6
+static int
b1dca6
+do_test (void)
b1dca6
+{
b1dca6
+  /* Verify what the audit modules have written.  This test assumes
b1dca6
+     that standard output has been redirected to a regular file.  */
b1dca6
+  FILE *fp = xfopen ("/dev/stdout", "r");
b1dca6
+
b1dca6
+  char *buffer = NULL;
b1dca6
+  size_t buffer_length = 0;
b1dca6
+  size_t line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  const char *message = "info: tst-auditlogmod-1.so loaded\n";
b1dca6
+  TEST_COMPARE_BLOB (message, strlen (message), buffer, line_length);
b1dca6
+
b1dca6
+  line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  message = "info: tst-auditlogmod-2.so loaded\n";
b1dca6
+  TEST_COMPARE_BLOB (message, strlen (message), buffer, line_length);
b1dca6
+
b1dca6
+  line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  message = "info: tst-auditlogmod-3.so loaded\n";
b1dca6
+  TEST_COMPARE_BLOB (message, strlen (message), buffer, line_length);
b1dca6
+
b1dca6
+  /* No more audit module output.  */
b1dca6
+  line_length = xgetline (&buffer, &buffer_length, fp);
b1dca6
+  TEST_COMPARE_BLOB ("", 0, buffer, line_length);
b1dca6
+
b1dca6
+  free (buffer);
b1dca6
+  xfclose (fp);
b1dca6
+  return 0;
b1dca6
+}
b1dca6
+
b1dca6
+#include <support/test-driver.c>
b1dca6
diff --git a/elf/tst-auditlogmod-1.c b/elf/tst-auditlogmod-1.c
b1dca6
new file mode 100644
b1dca6
index 0000000000000000..db9dac4c01a58dcd
b1dca6
--- /dev/null
b1dca6
+++ b/elf/tst-auditlogmod-1.c
b1dca6
@@ -0,0 +1,27 @@
b1dca6
+/* Audit module which logs that it was loaded.  Variant 1.
b1dca6
+   Copyright (C) 2020 Free Software Foundation, Inc.
b1dca6
+   This file is part of the GNU C Library.
b1dca6
+
b1dca6
+   The GNU C Library is free software; you can redistribute it and/or
b1dca6
+   modify it under the terms of the GNU Lesser General Public
b1dca6
+   License as published by the Free Software Foundation; either
b1dca6
+   version 2.1 of the License, or (at your option) any later version.
b1dca6
+
b1dca6
+   The GNU C Library is distributed in the hope that it will be useful,
b1dca6
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
b1dca6
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
b1dca6
+   Lesser General Public License for more details.
b1dca6
+
b1dca6
+   You should have received a copy of the GNU Lesser General Public
b1dca6
+   License along with the GNU C Library; if not, see
b1dca6
+   <https://www.gnu.org/licenses/>.  */
b1dca6
+
b1dca6
+#include <link.h>
b1dca6
+#include <support/support.h>
b1dca6
+
b1dca6
+unsigned int
b1dca6
+la_version (unsigned int v)
b1dca6
+{
b1dca6
+  write_message ("info: tst-auditlogmod-1.so loaded\n");
b1dca6
+  return LAV_CURRENT;
b1dca6
+}
b1dca6
diff --git a/elf/tst-auditlogmod-2.c b/elf/tst-auditlogmod-2.c
b1dca6
new file mode 100644
b1dca6
index 0000000000000000..871c22641c47fed0
b1dca6
--- /dev/null
b1dca6
+++ b/elf/tst-auditlogmod-2.c
b1dca6
@@ -0,0 +1,27 @@
b1dca6
+/* Audit module which logs that it was loaded.  Variant 2.
b1dca6
+   Copyright (C) 2020 Free Software Foundation, Inc.
b1dca6
+   This file is part of the GNU C Library.
b1dca6
+
b1dca6
+   The GNU C Library is free software; you can redistribute it and/or
b1dca6
+   modify it under the terms of the GNU Lesser General Public
b1dca6
+   License as published by the Free Software Foundation; either
b1dca6
+   version 2.1 of the License, or (at your option) any later version.
b1dca6
+
b1dca6
+   The GNU C Library is distributed in the hope that it will be useful,
b1dca6
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
b1dca6
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
b1dca6
+   Lesser General Public License for more details.
b1dca6
+
b1dca6
+   You should have received a copy of the GNU Lesser General Public
b1dca6
+   License along with the GNU C Library; if not, see
b1dca6
+   <https://www.gnu.org/licenses/>.  */
b1dca6
+
b1dca6
+#include <link.h>
b1dca6
+#include <support/support.h>
b1dca6
+
b1dca6
+unsigned int
b1dca6
+la_version (unsigned int v)
b1dca6
+{
b1dca6
+  write_message ("info: tst-auditlogmod-2.so loaded\n");
b1dca6
+  return LAV_CURRENT;
b1dca6
+}
b1dca6
diff --git a/elf/tst-auditlogmod-3.c b/elf/tst-auditlogmod-3.c
b1dca6
new file mode 100644
b1dca6
index 0000000000000000..da2ee19d4ab9e861
b1dca6
--- /dev/null
b1dca6
+++ b/elf/tst-auditlogmod-3.c
b1dca6
@@ -0,0 +1,27 @@
b1dca6
+/* Audit module which logs that it was loaded.  Variant 3.
b1dca6
+   Copyright (C) 2020 Free Software Foundation, Inc.
b1dca6
+   This file is part of the GNU C Library.
b1dca6
+
b1dca6
+   The GNU C Library is free software; you can redistribute it and/or
b1dca6
+   modify it under the terms of the GNU Lesser General Public
b1dca6
+   License as published by the Free Software Foundation; either
b1dca6
+   version 2.1 of the License, or (at your option) any later version.
b1dca6
+
b1dca6
+   The GNU C Library is distributed in the hope that it will be useful,
b1dca6
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
b1dca6
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
b1dca6
+   Lesser General Public License for more details.
b1dca6
+
b1dca6
+   You should have received a copy of the GNU Lesser General Public
b1dca6
+   License along with the GNU C Library; if not, see
b1dca6
+   <https://www.gnu.org/licenses/>.  */
b1dca6
+
b1dca6
+#include <link.h>
b1dca6
+#include <support/support.h>
b1dca6
+
b1dca6
+unsigned int
b1dca6
+la_version (unsigned int v)
b1dca6
+{
b1dca6
+  write_message ("info: tst-auditlogmod-3.so loaded\n");
b1dca6
+  return LAV_CURRENT;
b1dca6
+}