unofficial mirror of emacs-devel@gnu.org 
 help / color / mirror / code / Atom feed
* Re: boot-time: straighten code
       [not found] <4536176.VaOIPsP7d9@nimes>
@ 2023-08-13  2:49 ` Paul Eggert
  2023-08-13  3:26   ` Po Lu
                     ` (2 more replies)
  0 siblings, 3 replies; 24+ messages in thread
From: Paul Eggert @ 2023-08-13  2:49 UTC (permalink / raw)
  To: Bruno Haible; +Cc: bug-gnulib, Emacs Development

[-- Attachment #1: Type: text/plain, Size: 927 bytes --]

On 2023-08-11 14:49, Bruno Haible wrote:
> Paul: With this simplification, you may consider using the 'boot-time' module
> in Emacs. I bet that it produces a better result than Emacs' src/filelock.c
> on many platforms. (I haven't tested it, but I could test it if you give me
> a manual testing recipe.)

Thanks for doing all that. I installed the attached patch into Emacs 
master, which you should be able to test via:

	git clone https://git.savannah.gnu.org/git/emacs.git
	cd emacs
	./autogen.sh
	./configure
	make
	src/emacs

Please give it a try, especially on any MS-Windows platform you happen 
to have. I have tested only on Ubuntu 23.04 so far.

A simple way to test is to use Emacs to start editing a file (without 
saving) and then inspect the symbolic link .#* that Emacs uses as a lock 
file. The trailing digits of that link's contents should be the boot 
time. These symlinks are Emacs's only use of boot time.

[-- Attachment #2: 0001-Improve-boot-time-gathering.patch --]
[-- Type: text/x-patch, Size: 51156 bytes --]

From 5e736ca6ccfa131736ab0b3a298de2cb319e7dfb Mon Sep 17 00:00:00 2001
From: Paul Eggert <eggert@cs.ucla.edu>
Date: Sat, 12 Aug 2023 19:39:11 -0700
Subject: [PATCH] Improve boot-time gathering
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Simplify Emacs proper by using Gnulib’s boot-time module
instead of doing it all by hand.  This should port Emacs
better to obscurish hosts, as Bruno Haible has merged the
best of Emacs’s and Gnulib’s boot-time gathering.
* lib/boot-time-aux.h, lib/boot-time.c, lib/boot-time.h:
* lib/readutmp.h, m4/readutmp.m4: New files, copied from Gnulib.
* admin/merge-gnulib (GNULIB_MODULES): Add boot-time.
* configure.ac: Do not check for utmp.h;
the boot-time module now does this.
(BOOT_TIME_FILE): Remove; no longer used.
* lib/gnulib.mk.in, m4/gnulib-comp.m4: Regenerate.
* src/filelock.c [__FreeBSD__]: Do not include <sys/sysctl.h>.
[HAVE_UTMP_H]: Do not include utmp.h.
Include boot-time instead: boot-time does the work now.
(BOOT_TIME) [HAVE_ANDROID && !ANDROID_STUBIFY]: Don’t undef.
(WTMP_FILE): Don’t define.
(boot_time, boot_time_initialized, get_boot_time_1, get_boot_time):
Remove.
(get_boot_sec): New function that simply calls Gnulib get_boot_time.
(lock_file_1, current_lock_owner): Use get_boot_sec instead
of get_boot_time.
---
 admin/merge-gnulib  |   2 +-
 configure.ac        |  47 +------
 lib/boot-time-aux.h | 315 ++++++++++++++++++++++++++++++++++++++++++
 lib/boot-time.c     | 285 ++++++++++++++++++++++++++++++++++++++
 lib/boot-time.h     |  44 ++++++
 lib/gnulib.mk.in    |  11 ++
 lib/readutmp.h      | 325 ++++++++++++++++++++++++++++++++++++++++++++
 m4/gnulib-comp.m4   |   7 +
 m4/readutmp.m4      | 117 ++++++++++++++++
 src/filelock.c      | 172 ++---------------------
 10 files changed, 1116 insertions(+), 209 deletions(-)
 create mode 100644 lib/boot-time-aux.h
 create mode 100644 lib/boot-time.c
 create mode 100644 lib/boot-time.h
 create mode 100644 lib/readutmp.h
 create mode 100644 m4/readutmp.m4

diff --git a/admin/merge-gnulib b/admin/merge-gnulib
index 2a713beb01a..fe88d1106ae 100755
--- a/admin/merge-gnulib
+++ b/admin/merge-gnulib
@@ -26,7 +26,7 @@
 GNULIB_URL=https://git.savannah.gnu.org/git/gnulib.git
 
 GNULIB_MODULES='
-  alignasof alloca-opt binary-io byteswap c-ctype c-strcase
+  alignasof alloca-opt binary-io boot-time byteswap c-ctype c-strcase
   canonicalize-lgpl
   careadlinkat close-stream copy-file-range
   count-leading-zeros count-one-bits count-trailing-zeros
diff --git a/configure.ac b/configure.ac
index 35dfed247d0..0234a82b92f 100644
--- a/configure.ac
+++ b/configure.ac
@@ -2539,7 +2539,7 @@ AC_DEFUN
   sys/sysinfo.h
   coff.h pty.h
   sys/resource.h
-  sys/utsname.h pwd.h utmp.h util.h
+  sys/utsname.h pwd.h util.h
   sanitizer/lsan_interface.h
   sanitizer/asan_interface.h
   sanitizer/common_interface_defs.h])
@@ -2635,51 +2635,6 @@ AC_DEFUN
 fi
 AC_SUBST([AUTO_DEPEND])
 
-BOOT_TIME_FILE=
-AC_CACHE_CHECK([for old but post-boot file],
-  [emacs_cv_boot_time_file],
-  [AS_CASE([$opsys],
-     [gnu-linux],
-     [emacs_cv_boot_time_file=unknown
-      AS_IF([test $cross_compiling = no],
-	[# systemd puts it in /var/lib/systemd.
-	 # initscripts puts it in /var/lib/urandom (previously /var/lib).
-	 # Linux drivers/char/random.c before 2022-02-21 suggests /var/run.
-	 for file in \
-	     /var/lib/systemd/random-seed \
-	     /var/lib/urandom/random-seed \
-	     /var/lib/random-seed \
-	     /var/run/random-seed
-	 do
-	   test -f $file && { emacs_cv_boot_time_file=$file; break; }
-	 done])],
-     # This isn't perfect, as some systems might have the page file in
-     # another place.  Also, I suspect that the time stamp of that
-     # file might also change when Windows enlarges the file due to
-     # insufficient VM.  Still, this seems to be the most reliable
-     # way; the alternative (of using GetSystemTimes) won't work on
-     # laptops that hibernate, because the system clock is stopped
-     # then.  Other possibility would be to run "net statistics
-     # workstation" and parse the output, but that's gross.  So this
-     # should do; if the file is not there, the boot time will be
-     # returned as zero, and filelock.c already handles that.
-     [mingw32], [emacs_cv_boot_time_file=C:/pagefile.sys],
-     [*], [emacs_cv_boot_time_file=not-needed])])
-
-AS_CASE([$emacs_cv_boot_time_file],
-  [/*|*:*], [BOOT_TIME_FILE=\"$emacs_cv_boot_time_file\"],
-  [not-needed], [BOOT_TIME_FILE=],
-  [# Guess systemd if unknown.
-   # If guess is wrong, Emacs falls back on something else.
-   BOOT_TIME_FILE=\"/var/lib/systemd/random-seed\"])
-
-AS_IF([test -n "$BOOT_TIME_FILE"],
-  [AC_DEFINE_UNQUOTED([BOOT_TIME_FILE], [$BOOT_TIME_FILE],
-    [Name of file that, if it exists, postdates boot and predates
-     the first Emacs invocation; or a null pointer if no such file is known.
-     This file is used only on GNU/Linux and other systems
-     that lack the FreeBSD-style sysctl with KERN_BOOTTIME.])])
-
 #### Choose a window system.
 
 ## We leave window_system equal to none if
diff --git a/lib/boot-time-aux.h b/lib/boot-time-aux.h
new file mode 100644
index 00000000000..348611fc85c
--- /dev/null
+++ b/lib/boot-time-aux.h
@@ -0,0 +1,315 @@
+/* Auxiliary functions for determining the time when the machine last booted.
+   Copyright (C) 2023 Free Software Foundation, Inc.
+
+   This file 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 file 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <bruno@clisp.org>.  */
+
+#define SIZEOF(a) (sizeof(a)/sizeof(a[0]))
+
+#if defined __linux__ || defined __ANDROID__
+
+/* Store the uptime counter, as managed by the Linux kernel, in *P_UPTIME.
+   Return 0 upon success, -1 upon failure.  */
+_GL_ATTRIBUTE_MAYBE_UNUSED
+static int
+get_linux_uptime (struct timespec *p_uptime)
+{
+  /* The clock_gettime facility returns the uptime with a resolution of 1 µsec.
+     It is available with glibc >= 2.14, Android, or musl libc.
+     In glibc < 2.17 it required linking with librt.  */
+# if !defined __GLIBC__ || 2 < __GLIBC__ + (17 <= __GLIBC_MINOR__)
+  if (clock_gettime (CLOCK_BOOTTIME, p_uptime) >= 0)
+    return 0;
+# endif
+
+  /* /proc/uptime contains the uptime with a resolution of 0.01 sec.
+     But it does not have read permissions on Android.  */
+# if !defined __ANDROID__
+  FILE *fp = fopen ("/proc/uptime", "re");
+  if (fp != NULL)
+    {
+      char buf[32 + 1];
+      size_t n = fread (buf, 1, sizeof (buf) - 1, fp);
+      fclose (fp);
+      if (n > 0)
+        {
+          buf[n] = '\0';
+          /* buf now contains two values: the uptime and the idle time.  */
+          time_t s = 0;
+          char *p;
+          for (p = buf; '0' <= *p && *p <= '9'; p++)
+            s = 10 * s + (*p - '0');
+          if (buf < p)
+            {
+              long ns = 0;
+              if (*p++ == '.')
+                for (int i = 0; i < 9; i++)
+                  ns = 10 * ns + ('0' <= *p && *p <= '9' ? *p++ - '0' : 0);
+              p_uptime->tv_sec = s;
+              p_uptime->tv_nsec = ns;
+              return 0;
+            }
+        }
+    }
+# endif
+
+  /* The sysinfo call returns the uptime with a resolution of 1 sec only.  */
+  struct sysinfo info;
+  if (sysinfo (&info) >= 0)
+    {
+      p_uptime->tv_sec = info.uptime;
+      p_uptime->tv_nsec = 0;
+      return 0;
+    }
+
+  return -1;
+}
+
+#endif
+
+#if defined __linux__ && !defined __ANDROID__
+
+static int
+get_linux_boot_time_fallback (struct timespec *p_boot_time)
+{
+  /* On Alpine Linux, UTMP_FILE is not filled.  It is always empty.
+     So, get the time stamp of a file that gets touched only during the
+     boot process.  */
+
+  const char * const boot_touched_files[] =
+    {
+      "/var/lib/systemd/random-seed", /* seen on distros with systemd */
+      "/var/run/utmp",                /* seen on distros with OpenRC */
+      "/var/lib/random-seed"          /* seen on older distros */
+    };
+  for (idx_t i = 0; i < SIZEOF (boot_touched_files); i++)
+    {
+      const char *filename = boot_touched_files[i];
+      struct stat statbuf;
+      if (stat (filename, &statbuf) >= 0)
+        {
+          *p_boot_time = get_stat_mtime (&statbuf);
+          return 0;
+        }
+    }
+  return -1;
+}
+
+/* The following approach is only usable as a fallback, because it is of
+   the form
+     boot_time = (time now) - (kernel's ktime_get_boottime[_ts64] ())
+   and therefore produces wrong values after the date has been bumped in the
+   running system, which happens frequently if the system is running in a
+   virtual machine and this VM has been put into "saved" or "sleep" state
+   and then resumed.  */
+static int
+get_linux_boot_time_final_fallback (struct timespec *p_boot_time)
+{
+  struct timespec uptime;
+  if (get_linux_uptime (&uptime) >= 0)
+    {
+      struct timespec result;
+# if !defined __GLIBC__ || 2 < __GLIBC__ + (16 <= __GLIBC_MINOR__)
+      /* Better than:
+      if (0 <= clock_gettime (CLOCK_REALTIME, &result))
+         because timespec_get does not need -lrt in glibc 2.16.
+      */
+      if (! timespec_get (&result, TIME_UTC))
+        return -1;
+#  else
+      /* Fall back on lower-res approach that does not need -lrt.
+         This is good enough; on these hosts UPTIME is even lower-res.  */
+      struct timeval tv;
+      int r = gettimeofday (&tv, NULL);
+      if (r < 0)
+        return r;
+      result.tv_sec = tv.tv_sec;
+      result.tv_nsec = tv.tv_usec * 1000;
+#  endif
+
+      if (result.tv_nsec < uptime.tv_nsec)
+        {
+          result.tv_nsec += 1000000000;
+          result.tv_sec -= 1;
+        }
+      result.tv_sec -= uptime.tv_sec;
+      result.tv_nsec -= uptime.tv_nsec;
+      *p_boot_time = result;
+      return 0;
+    }
+  return -1;
+}
+
+#endif
+
+#if defined __ANDROID__
+
+static int
+get_android_boot_time (struct timespec *p_boot_time)
+{
+  /* On Android, there is no /var, and normal processes don't have access
+     to system files.  Therefore use the kernel's uptime counter, although
+     it produces wrong values after the date has been bumped in the running
+     system.  */
+  struct timespec uptime;
+  if (get_linux_uptime (&uptime) >= 0)
+    {
+      struct timespec result;
+      if (clock_gettime (CLOCK_REALTIME, &result) >= 0)
+        {
+          if (result.tv_nsec < uptime.tv_nsec)
+            {
+              result.tv_nsec += 1000000000;
+              result.tv_sec -= 1;
+            }
+          result.tv_sec -= uptime.tv_sec;
+          result.tv_nsec -= uptime.tv_nsec;
+          *p_boot_time = result;
+          return 0;
+        }
+    }
+  return -1;
+}
+
+#endif
+
+#if defined __OpenBSD__
+
+static int
+get_openbsd_boot_time (struct timespec *p_boot_time)
+{
+  /* On OpenBSD, UTMP_FILE is not filled.  It contains only dummy entries.
+     So, get the time stamp of a file that gets touched only during the
+     boot process.  */
+  const char * const boot_touched_files[] =
+    {
+      "/var/db/host.random",
+      "/var/run/utmp"
+    };
+  for (idx_t i = 0; i < SIZEOF (boot_touched_files); i++)
+    {
+      const char *filename = boot_touched_files[i];
+      struct stat statbuf;
+      if (stat (filename, &statbuf) >= 0)
+        {
+          *p_boot_time = get_stat_mtime (&statbuf);
+          return 0;
+        }
+    }
+  return -1;
+}
+
+#endif
+
+#if HAVE_SYS_SYSCTL_H && HAVE_SYSCTL \
+    && defined CTL_KERN && defined KERN_BOOTTIME \
+    && !defined __minix
+/* macOS, FreeBSD, GNU/kFreeBSD, NetBSD, OpenBSD */
+/* On Minix 3.3 this sysctl produces garbage results.  Therefore avoid it.  */
+
+/* The following approach is only usable as a fallback, because it produces
+   wrong values after the date has been bumped in the running system, which
+   happens frequently if the system is running in a virtual machine and this
+   VM has been put into "saved" or "sleep" state and then resumed.  */
+static int
+get_bsd_boot_time_final_fallback (struct timespec *p_boot_time)
+{
+  static int request[2] = { CTL_KERN, KERN_BOOTTIME };
+  struct timeval result;
+  size_t result_len = sizeof result;
+
+  if (sysctl (request, 2, &result, &result_len, NULL, 0) >= 0)
+    {
+      p_boot_time->tv_sec = result.tv_sec;
+      p_boot_time->tv_nsec = result.tv_usec * 1000;
+      return 0;
+    }
+  return -1;
+}
+
+#endif
+
+#if defined __HAIKU__
+
+static int
+get_haiku_boot_time (struct timespec *p_boot_time)
+{
+  /* On Haiku, /etc/utmp does not exist.  During boot,
+       1. the current time is restored, but possibly with a wrong time zone,
+          that is, with an offset of a few hours,
+       2. some symlinks and files get created,
+       3. the various devices are brought up, in particular the network device,
+       4. the correct date and time is set,
+       5. some more device nodes get created.
+     The boot time can be retrieved by looking at a directory created during
+     phase 5, such as /dev/input.  */
+  const char * const boot_touched_file = "/dev/input";
+  struct stat statbuf;
+  if (stat (boot_touched_file, &statbuf) >= 0)
+    {
+      *p_boot_time = get_stat_mtime (&statbuf);
+      return 0;
+    }
+  return -1;
+}
+
+#endif
+
+#if HAVE_OS_H /* BeOS, Haiku */
+
+/* The following approach is only usable as a fallback, because it produces
+   wrong values after the date has been bumped in the running system, which
+   happens frequently if the system is running in a virtual machine and this
+   VM has been put into "saved" or "sleep" state and then resumed.  */
+static int
+get_haiku_boot_time_final_fallback (struct timespec *p_boot_time)
+{
+  system_info si;
+
+  get_system_info (&si);
+  p_boot_time->tv_sec = si.boot_time / 1000000;
+  p_boot_time->tv_nsec = (si.boot_time % 1000000) * 1000;
+  return 0;
+}
+
+#endif
+
+#if defined __CYGWIN__ || defined _WIN32
+
+static int
+get_windows_boot_time (struct timespec *p_boot_time)
+{
+  /* On Cygwin, /var/run/utmp is empty.
+     On native Windows, <utmpx.h> and <utmp.h> don't exist.
+     Instead, on Windows, the boot time can be retrieved by looking at the
+     time stamp of a file that (normally) gets touched only during the boot
+     process, namely C:\pagefile.sys.  */
+  const char * const boot_touched_file =
+    #if defined __CYGWIN__ && !defined _WIN32
+    "/cygdrive/c/pagefile.sys"
+    #else
+    "C:\\pagefile.sys"
+    #endif
+    ;
+  struct stat statbuf;
+  if (stat (boot_touched_file, &statbuf) >= 0)
+    {
+      *p_boot_time = get_stat_mtime (&statbuf);
+      return 0;
+    }
+  return -1;
+}
+
+#endif
diff --git a/lib/boot-time.c b/lib/boot-time.c
new file mode 100644
index 00000000000..d813bfa5825
--- /dev/null
+++ b/lib/boot-time.c
@@ -0,0 +1,285 @@
+/* Determine the time when the machine last booted.
+   Copyright (C) 2023 Free Software Foundation, Inc.
+
+   This file 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 file 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <bruno@clisp.org>.  */
+
+#include <config.h>
+
+/* Specification.  */
+#include "boot-time.h"
+
+#include <stddef.h>
+#include <stdio.h>
+#include <string.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+
+#if defined __linux__ || defined __ANDROID__
+# include <sys/sysinfo.h>
+# include <time.h>
+#endif
+
+#if HAVE_SYS_SYSCTL_H && !defined __minix
+# if HAVE_SYS_PARAM_H
+#  include <sys/param.h>
+# endif
+# include <sys/sysctl.h>
+#endif
+
+#if HAVE_OS_H
+# include <OS.h>
+#endif
+
+#include "idx.h"
+#include "readutmp.h"
+#include "stat-time.h"
+
+/* Each of the FILE streams in this file is only used in a single thread.  */
+#include "unlocked-io.h"
+
+/* Some helper functions.  */
+#include "boot-time-aux.h"
+
+/* The following macros describe the 'struct UTMP_STRUCT_NAME',
+   *not* 'struct gl_utmp'.  */
+#undef UT_USER
+
+/* Accessor macro for the member named ut_user or ut_name.  */
+#if (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_NAME \
+     : HAVE_UTMP_H && HAVE_STRUCT_UTMP_UT_NAME)
+# define UT_USER(UT) ((UT)->ut_name)
+#else
+# define UT_USER(UT) ((UT)->ut_user)
+#endif
+
+#if !HAVE_UTMPX_H && HAVE_UTMP_H && defined UTMP_NAME_FUNCTION && !HAVE_DECL_GETUTENT
+struct utmp *getutent (void);
+#endif
+
+#if defined __linux__ || HAVE_UTMPX_H || HAVE_UTMP_H || defined __CYGWIN__ || defined _WIN32
+
+static int
+get_boot_time_uncached (struct timespec *p_boot_time)
+{
+  struct timespec found_boot_time = {0};
+
+# if (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_TYPE : HAVE_STRUCT_UTMP_UT_TYPE)
+
+  /* Try to find the boot time in the /var/run/utmp file.  */
+
+#  if defined UTMP_NAME_FUNCTION /* glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, IRIX, Solaris, Cygwin, Android */
+
+  /* Ignore the return value for now.
+     Solaris' utmpname returns 1 upon success -- which is contrary
+     to what the GNU libc version does.  In addition, older GNU libc
+     versions are actually void.   */
+  UTMP_NAME_FUNCTION ((char *) UTMP_FILE);
+
+  SET_UTMP_ENT ();
+
+#   if (defined __linux__ && !defined __ANDROID__) || defined __minix
+  /* Timestamp of the "runlevel" entry, if any.  */
+  struct timespec runlevel_ts = {0};
+#   endif
+
+  void const *entry;
+
+  while ((entry = GET_UTMP_ENT ()) != NULL)
+    {
+      struct UTMP_STRUCT_NAME const *ut = (struct UTMP_STRUCT_NAME const *) entry;
+
+      struct timespec ts =
+        #if (HAVE_UTMPX_H ? 1 : HAVE_STRUCT_UTMP_UT_TV)
+        { .tv_sec = ut->ut_tv.tv_sec, .tv_nsec = ut->ut_tv.tv_usec * 1000 };
+        #else
+        { .tv_sec = ut->ut_time, .tv_nsec = 0 };
+        #endif
+
+      if (ut->ut_type == BOOT_TIME)
+        found_boot_time = ts;
+
+#   if defined __linux__ && !defined __ANDROID__
+      if (memcmp (UT_USER (ut), "runlevel", strlen ("runlevel") + 1) == 0
+          && memcmp (ut->ut_line, "~", strlen ("~") + 1) == 0)
+        runlevel_ts = ts;
+#   endif
+#   if defined __minix
+      if (UT_USER (ut)[0] == '\0'
+          && memcmp (ut->ut_line, "run-level ", strlen ("run-level ")) == 0)
+        runlevel_ts = ts;
+#   endif
+    }
+
+  END_UTMP_ENT ();
+
+#   if defined __linux__ && !defined __ANDROID__
+  /* On Raspbian, which runs on hardware without a real-time clock, during boot,
+       1. the clock gets set to 1970-01-01 00:00:00,
+       2. an entry gets written into /var/run/utmp, with ut_type = BOOT_TIME,
+          ut_user = "reboot", ut_line = "~", time = 1970-01-01 00:00:05 or so,
+       3. the clock gets set to a correct value through NTP,
+       4. an entry gets written into /var/run/utmp, with
+          ut_user = "runlevel", ut_line = "~", time = correct value.
+     In this case, get the time from the "runlevel" entry.  */
+
+  /* Workaround for Raspbian:  */
+  if (found_boot_time.tv_sec <= 60 && runlevel_ts.tv_sec != 0)
+    found_boot_time = runlevel_ts;
+  if (found_boot_time.tv_sec == 0)
+    {
+      /* Workaround for Alpine Linux:  */
+      get_linux_boot_time_fallback (&found_boot_time);
+    }
+#   endif
+
+#   if defined __ANDROID__
+  if (found_boot_time.tv_sec == 0)
+    {
+      /* Workaround for Android:  */
+      get_android_boot_time (&found_boot_time);
+    }
+#   endif
+
+#   if defined __minix
+  /* On Minix, during boot,
+       1. an entry gets written into /var/run/utmp, with ut_type = BOOT_TIME,
+          ut_user = "", ut_line = "system boot", time = 1970-01-01 00:00:00,
+       2. an entry gets written into /var/run/utmp, with
+          ut_user = "", ut_line = "run-level m", time = correct value.
+     In this case, copy the time from the "run-level m" entry to the
+     "system boot" entry.  */
+  if (found_boot_time.tv_sec <= 60 && runlevel_ts.tv_sec != 0)
+    found_boot_time = runlevel_ts;
+#   endif
+
+#  else /* HP-UX, Haiku */
+
+  FILE *f = fopen (UTMP_FILE, "re");
+
+  if (f != NULL)
+    {
+      for (;;)
+        {
+          struct UTMP_STRUCT_NAME ut;
+
+          if (fread (&ut, sizeof ut, 1, f) == 0)
+            break;
+
+          struct timespec ts =
+            #if (HAVE_UTMPX_H ? 1 : HAVE_STRUCT_UTMP_UT_TV)
+            { .tv_sec = ut.ut_tv.tv_sec, .tv_nsec = ut.ut_tv.tv_usec * 1000 };
+            #else
+            { .tv_sec = ut.ut_time, .tv_nsec = 0 };
+            #endif
+
+          if (ut.ut_type == BOOT_TIME)
+            found_boot_time = ts;
+        }
+
+      fclose (f);
+    }
+
+#  endif
+
+#  if defined __linux__ && !defined __ANDROID__
+  if (found_boot_time.tv_sec == 0)
+    {
+      get_linux_boot_time_final_fallback (&found_boot_time);
+    }
+#  endif
+
+# else /* old FreeBSD, OpenBSD, native Windows */
+
+#  if defined __OpenBSD__
+  /* Workaround for OpenBSD:  */
+  get_openbsd_boot_time (&found_boot_time);
+#  endif
+
+# endif
+
+# if HAVE_SYS_SYSCTL_H && HAVE_SYSCTL \
+     && defined CTL_KERN && defined KERN_BOOTTIME \
+     && !defined __minix
+  if (found_boot_time.tv_sec == 0)
+    {
+      get_bsd_boot_time_final_fallback (&found_boot_time);
+    }
+# endif
+
+# if defined __HAIKU__
+  if (found_boot_time.tv_sec == 0)
+    {
+      get_haiku_boot_time (&found_boot_time);
+    }
+# endif
+
+# if HAVE_OS_H
+  if (found_boot_time.tv_sec == 0)
+    {
+      get_haiku_boot_time_final_fallback (&found_boot_time);
+    }
+# endif
+
+# if defined __CYGWIN__ || defined _WIN32
+  if (found_boot_time.tv_sec == 0)
+    {
+      /* Workaround for Windows:  */
+      get_windows_boot_time (&found_boot_time);
+    }
+# endif
+
+  if (found_boot_time.tv_sec != 0)
+    {
+      *p_boot_time = found_boot_time;
+      return 0;
+    }
+  else
+    return -1;
+}
+
+int
+get_boot_time (struct timespec *p_boot_time)
+{
+  /* Cache the result from get_boot_time_uncached.  */
+  static int volatile cached_result = -1;
+  static struct timespec volatile cached_boot_time;
+
+  if (cached_result < 0)
+    {
+      struct timespec boot_time;
+      int result = get_boot_time_uncached (&boot_time);
+      cached_boot_time = boot_time;
+      cached_result = result;
+    }
+
+  if (cached_result == 0)
+    {
+      *p_boot_time = cached_boot_time;
+      return 0;
+    }
+  else
+    return -1;
+}
+
+#else
+
+int
+get_boot_time (struct timespec *p_boot_time)
+{
+  return -1;
+}
+
+#endif
diff --git a/lib/boot-time.h b/lib/boot-time.h
new file mode 100644
index 00000000000..401e854adbb
--- /dev/null
+++ b/lib/boot-time.h
@@ -0,0 +1,44 @@
+/* Determine the time when the machine last booted.
+   Copyright (C) 2023 Free Software Foundation, Inc.
+
+   This file 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 file 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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by Bruno Haible <bruno@clisp.org>.  */
+
+#ifndef _BOOT_TIME_H
+#define _BOOT_TIME_H
+
+#include <time.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/* Store the approximate time when the machine last booted in *P_BOOT_TIME,
+   and return 0.  If it cannot be determined, return -1.
+
+   This function is not multithread-safe, since on many platforms it
+   invokes the functions setutxent, getutxent, endutxent.  These
+   functions are needed because they may lock FILE (so that we don't
+   read garbage when a concurrent process writes to FILE), but their
+   drawback is that they have a common global state.  */
+extern int get_boot_time (struct timespec *p_boot_time);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* _BOOT_TIME_H */
diff --git a/lib/gnulib.mk.in b/lib/gnulib.mk.in
index 785bdc70c5c..3b33f39f73b 100644
--- a/lib/gnulib.mk.in
+++ b/lib/gnulib.mk.in
@@ -76,6 +76,7 @@
 #  alignasof \
 #  alloca-opt \
 #  binary-io \
+#  boot-time \
 #  byteswap \
 #  c-ctype \
 #  c-strcase \
@@ -1601,6 +1602,16 @@ libgnu_a_SOURCES += binary-io.h binary-io.c
 endif
 ## end   gnulib module binary-io
 
+## begin gnulib module boot-time
+ifeq (,$(OMIT_GNULIB_MODULE_boot-time))
+
+libgnu_a_SOURCES += boot-time.c
+
+EXTRA_DIST += boot-time-aux.h boot-time.h readutmp.h
+
+endif
+## end   gnulib module boot-time
+
 ## begin gnulib module byteswap
 ifeq (,$(OMIT_GNULIB_MODULE_byteswap))
 
diff --git a/lib/readutmp.h b/lib/readutmp.h
new file mode 100644
index 00000000000..1cf588d265b
--- /dev/null
+++ b/lib/readutmp.h
@@ -0,0 +1,325 @@
+/* Declarations for GNU's read utmp module.
+
+   Copyright (C) 1992-2007, 2009-2023 Free Software Foundation, Inc.
+
+   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 <https://www.gnu.org/licenses/>.  */
+
+/* Written by jla; revised by djm */
+
+#ifndef __READUTMP_H__
+#define __READUTMP_H__
+
+/* This file uses _GL_ATTRIBUTE_MALLOC, _GL_ATTRIBUTE_RETURNS_NONNULL,
+   HAVE_UTMP_H, HAVE_UTMPX_H, HAVE_STRUCT_UTMP_*, HAVE_STRUCT_UTMPX_*,
+   HAVE_UTMPNAME, HAVE_UTMPXNAME.  */
+#if !_GL_CONFIG_H_INCLUDED
+# error "Please include config.h first."
+#endif
+
+#include "idx.h"
+
+#include <stdlib.h>
+#include <sys/types.h>
+#include <time.h>
+
+/* AIX 4.3.3 has both utmp.h and utmpx.h, but only struct utmp
+   has the ut_exit member.  */
+#if (HAVE_UTMPX_H && HAVE_UTMP_H && HAVE_STRUCT_UTMP_UT_EXIT \
+     && ! HAVE_STRUCT_UTMPX_UT_EXIT)
+# undef HAVE_UTMPX_H
+#endif
+
+/* HPUX 10.20 needs utmp.h, for the definition of e.g., UTMP_FILE.  */
+#if HAVE_UTMP_H
+# include <utmp.h>
+#endif
+
+/* Needed for BOOT_TIME and USER_PROCESS.  */
+#if HAVE_UTMPX_H
+# if defined _THREAD_SAFE && defined UTMP_DATA_INIT
+    /* When including both utmp.h and utmpx.h on AIX 4.3, with _THREAD_SAFE
+       defined, work around the duplicate struct utmp_data declaration.  */
+#  define utmp_data gl_aix_4_3_workaround_utmp_data
+# endif
+# include <utmpx.h>
+#endif
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/* Type of entries returned by read_utmp on all platforms.  */
+struct gl_utmp
+{
+  /* All 'char *' here are of arbitrary length and point to storage
+     with lifetime equal to that of this struct.  */
+  char *ut_user;                /* User name */
+  char *ut_id;                  /* Session ID */
+  char *ut_line;                /* seat / device */
+  char *ut_host;                /* for remote sessions: user@host or host,
+                                   for local sessions: the X11 display :N */
+  struct timespec ut_ts;        /* time */
+  pid_t ut_pid;                 /* process ID of ? */
+  pid_t ut_session;             /* process ID of session leader */
+  short ut_type;                /* BOOT_TIME, USER_PROCESS, or other */
+  struct { int e_termination; int e_exit; } ut_exit;
+};
+
+/* The following types, macros, and constants describe the 'struct gl_utmp'.  */
+#define UT_USER(UT) ((UT)->ut_user)
+#define UT_TIME_MEMBER(UT) ((UT)->ut_ts.tv_sec)
+#define UT_PID(UT) ((UT)->ut_pid)
+#define UT_TYPE_EQ(UT, V) ((UT)->ut_type == (V))
+#define UT_TYPE_NOT_DEFINED 0
+#define UT_EXIT_E_TERMINATION(UT) ((UT)->ut_exit.e_termination)
+#define UT_EXIT_E_EXIT(UT) ((UT)->ut_exit.e_exit)
+
+/* Type of entry returned by read_utmp().  */
+typedef struct gl_utmp STRUCT_UTMP;
+
+/* Size of the UT_USER (ut) member, or -1 if unbounded.  */
+enum { UT_USER_SIZE = -1 };
+
+/* Size of the ut->ut_id member, or -1 if unbounded.  */
+enum { UT_ID_SIZE = -1 };
+
+/* Size of the ut->ut_line member, or -1 if unbounded.  */
+enum { UT_LINE_SIZE = -1 };
+
+/* Size of the ut->ut_host member, or -1 if unbounded.  */
+enum { UT_HOST_SIZE = -1 };
+
+
+/* When read_utmp accesses a file (as opposed to fetching the information
+   from systemd), it uses the following low-level types and macros.
+   Keep them here, rather than moving them into readutmp.c, for backward
+   compatibility.  */
+
+#if HAVE_UTMPX_H
+
+/* <utmpx.h> defines 'struct utmpx' with the following fields:
+
+     Field        Type                       Platforms
+     ----------   ------                     ---------
+   ⎡ ut_user      char[]                     glibc, musl, macOS, FreeBSD, AIX, HP-UX, IRIX, Solaris, Cygwin
+   ⎣ ut_name      char[]                     NetBSD, Minix
+     ut_id        char[]                     glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+     ut_line      char[]                     glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+     ut_pid       pid_t                      glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+     ut_type      short                      glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+   ⎡ ut_tv        struct                     glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+   ⎢              { tv_sec; tv_usec; }
+   ⎣ ut_time      time_t                     Cygwin
+     ut_host      char[]                     glibc, musl, macOS, FreeBSD, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+     ut_exit      struct                     glibc, musl, NetBSD, Minix, HP-UX, IRIX, Solaris
+                  { e_termination; e_exit; }
+     ut_session   [long] int                 glibc, musl, NetBSD, Minix, IRIX, Solaris
+   ⎡ ut_addr      [long] int                 HP-UX, Cygwin
+   ⎢ ut_addr_v6   [u]int[4]                  glibc, musl
+   ⎣ ut_ss        struct sockaddr_storage    NetBSD, Minix
+ */
+
+# if __GLIBC__ && _TIME_BITS == 64
+/* This is a near-copy of glibc's struct utmpx, which stops working
+   after the year 2038.  Unlike the glibc version, struct utmpx32
+   describes the file format even if time_t is 64 bits.  */
+struct utmpx32
+{
+  short int ut_type;               /* Type of login.  */
+  pid_t ut_pid;                    /* Process ID of login process.  */
+  char ut_line[__UT_LINESIZE];     /* Devicename.  */
+  char ut_id[4];                   /* Inittab ID.  */
+  char ut_user[__UT_USERSIZE];     /* Username.  */
+  char ut_host[__UT_HOSTSIZE];     /* Hostname for remote login. */
+  struct __exit_status ut_exit;    /* Exit status of a process marked
+                                      as DEAD_PROCESS.  */
+  /* The fields ut_session and ut_tv must be the same size when compiled
+     32- and 64-bit.  This allows files and shared memory to be shared
+     between 32- and 64-bit applications.  */
+  int ut_session;                  /* Session ID, used for windowing.  */
+  struct
+  {
+    /* Seconds.  Unsigned not signed, as glibc did not exist before 1970,
+       and if the format is still in use after 2038 its timestamps
+       will surely have the sign bit on.  This hack stops working
+       at 2106-02-07 06:28:16 UTC.  */
+    unsigned int tv_sec;
+    int tv_usec;                   /* Microseconds.  */
+  } ut_tv;                         /* Time entry was made.  */
+  int ut_addr_v6[4];               /* Internet address of remote host.  */
+  char ut_reserved[20];            /* Reserved for future use.  */
+};
+#  define UTMP_STRUCT_NAME utmpx32
+# else
+#  define UTMP_STRUCT_NAME utmpx
+# endif
+# define SET_UTMP_ENT setutxent
+# define GET_UTMP_ENT getutxent
+# define END_UTMP_ENT endutxent
+# ifdef HAVE_UTMPXNAME /* glibc, musl, macOS, NetBSD, Minix, IRIX, Solaris, Cygwin */
+#  define UTMP_NAME_FUNCTION utmpxname
+# elif defined UTXDB_ACTIVE /* FreeBSD */
+#  define UTMP_NAME_FUNCTION(x) setutxdb (UTXDB_ACTIVE, x)
+# endif
+
+#elif HAVE_UTMP_H
+
+/* <utmp.h> defines 'struct utmp' with the following fields:
+
+     Field        Type                       Platforms
+     ----------   ------                     ---------
+   ⎡ ut_user      char[]                     glibc, musl, AIX, HP-UX, IRIX, Solaris, Cygwin, Android
+   ⎣ ut_name      char[]                     macOS, old FreeBSD, NetBSD, OpenBSD, Minix
+     ut_id        char[]                     glibc, musl, AIX, HP-UX, IRIX, Solaris, Cygwin, Android
+     ut_line      char[]                     glibc, musl, macOS, old FreeBSD, NetBSD, OpenBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin, Android
+     ut_pid       pid_t                      glibc, musl, AIX, HP-UX, IRIX, Solaris, Cygwin, Android
+     ut_type      short                      glibc, musl, AIX, HP-UX, IRIX, Solaris, Cygwin, Android
+   ⎡ ut_tv        struct                     glibc, musl, Android
+   ⎢              { tv_sec; tv_usec; }
+   ⎣ ut_time      time_t                     macOS, old FreeBSD, NetBSD, OpenBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin
+     ut_host      char[]                     glibc, musl, macOS, old FreeBSD, NetBSD, OpenBSD, Minix, AIX, HP-UX, Cygwin, Android
+     ut_exit      struct                     glibc, musl, AIX, HP-UX, IRIX, Solaris, Android
+                  { e_termination; e_exit; }
+     ut_session   [long] int                 glibc, musl, Android
+   ⎡ ut_addr      [long] int                 HP-UX, Cygwin
+   ⎣ ut_addr_v6   [u]int[4]                  glibc, musl, Android
+ */
+
+# define UTMP_STRUCT_NAME utmp
+# define SET_UTMP_ENT setutent
+# define GET_UTMP_ENT getutent
+# define END_UTMP_ENT endutent
+# ifdef HAVE_UTMPNAME /* glibc, musl, NetBSD, Minix, AIX, HP-UX, IRIX, Solaris, Cygwin, Android */
+#  define UTMP_NAME_FUNCTION utmpname
+# endif
+
+#endif
+
+/* Evaluates to 1 if gl_utmp's ut_id field may ever have a non-zero value.  */
+#define HAVE_STRUCT_XTMP_UT_ID \
+  (READUTMP_USE_SYSTEMD \
+   || (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_ID : HAVE_STRUCT_UTMP_UT_ID))
+
+/* Evaluates to 1 if gl_utmp's ut_pid field may ever have a non-zero value.  */
+#define HAVE_STRUCT_XTMP_UT_PID \
+  (READUTMP_USE_SYSTEMD \
+   || (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_PID : HAVE_STRUCT_UTMP_UT_PID))
+
+/* Evaluates to 1 if gl_utmp's ut_host field may ever be non-empty.  */
+#define HAVE_STRUCT_XTMP_UT_HOST \
+  (READUTMP_USE_SYSTEMD \
+   || (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_HOST : HAVE_STRUCT_UTMP_UT_HOST))
+
+/* Definition of UTMP_FILE.
+   On glibc systems, UTMP_FILE is "/var/run/utmp".  */
+#if !defined UTMP_FILE && defined _PATH_UTMP
+# define UTMP_FILE _PATH_UTMP
+#endif
+#ifdef UTMPX_FILE /* Solaris, SysVr4 */
+# undef UTMP_FILE
+# define UTMP_FILE UTMPX_FILE
+#endif
+#ifndef UTMP_FILE
+# define UTMP_FILE "/etc/utmp"
+#endif
+
+/* Definition of WTMP_FILE.
+   On glibc systems, UTMP_FILE is "/var/log/wtmp".  */
+#if !defined WTMP_FILE && defined _PATH_WTMP
+# define WTMP_FILE _PATH_WTMP
+#endif
+#ifdef WTMPX_FILE /* Solaris, SysVr4 */
+# undef WTMP_FILE
+# define WTMP_FILE WTMPX_FILE
+#endif
+#ifndef WTMP_FILE
+# define WTMP_FILE "/etc/wtmp"
+#endif
+
+/* Some platforms, such as OpenBSD, don't have an ut_type field and don't have
+   the BOOT_TIME and USER_PROCESS macros.  But we want to support them in
+   'struct gl_utmp'.  */
+#if !(HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_TYPE : HAVE_STRUCT_UTMP_UT_TYPE)
+# define BOOT_TIME 2
+# define USER_PROCESS 0
+#endif
+
+/* Macros that test (UT)->ut_type.  */
+#ifdef BOOT_TIME
+# define UT_TYPE_BOOT_TIME(UT) UT_TYPE_EQ (UT, BOOT_TIME)
+#else
+# define UT_TYPE_BOOT_TIME(UT) 0
+#endif
+#ifdef USER_PROCESS
+# define UT_TYPE_USER_PROCESS(UT) UT_TYPE_EQ (UT, USER_PROCESS)
+#else
+# define UT_TYPE_USER_PROCESS(UT) 0
+#endif
+
+/* Determines whether an entry *UT corresponds to a user process.  */
+#define IS_USER_PROCESS(UT)                                    \
+  (UT_USER (UT)[0]                                             \
+   && (UT_TYPE_USER_PROCESS (UT)                               \
+       || (UT_TYPE_NOT_DEFINED && UT_TIME_MEMBER (UT) != 0)))
+
+/* Define if read_utmp is not just a dummy.  */
+#if READUTMP_USE_SYSTEMD || HAVE_UTMPX_H || HAVE_UTMP_H || defined __CYGWIN__ || defined _WIN32
+# define READ_UTMP_SUPPORTED 1
+#endif
+
+/* Options for read_utmp.  */
+enum
+  {
+    READ_UTMP_CHECK_PIDS   = 1,
+    READ_UTMP_USER_PROCESS = 2,
+    READ_UTMP_BOOT_TIME    = 4,
+    READ_UTMP_NO_BOOT_TIME = 8
+  };
+
+/* Return a copy of (UT)->ut_user, without trailing spaces,
+   as a freshly allocated string.  */
+char *extract_trimmed_name (const STRUCT_UTMP *ut)
+  _GL_ATTRIBUTE_MALLOC _GL_ATTRIBUTE_DEALLOC_FREE
+  _GL_ATTRIBUTE_RETURNS_NONNULL;
+
+/* Read the utmp entries corresponding to file FILE into freshly-
+   malloc'd storage, set *UTMP_BUF to that pointer, set *N_ENTRIES to
+   the number of entries, and return zero.  If there is any error,
+   return -1, setting errno, and don't modify the parameters.
+   A good candidate for FILE is UTMP_FILE.
+   If OPTIONS & READ_UTMP_CHECK_PIDS is nonzero, omit entries whose
+   process-IDs do not currently exist.
+   If OPTIONS & READ_UTMP_USER_PROCESS is nonzero, omit entries which
+   do not correspond to a user process.
+   If OPTIONS & READ_UTMP_BOOT_TIME is nonzero, omit all entries except
+   the one that contains the boot time.
+   If OPTIONS & READ_UTMP_NO_BOOT_TIME is nonzero, omit the boot time
+   entries.
+
+   This function is not multithread-safe, since on many platforms it
+   invokes the functions setutxent, getutxent, endutxent.  These
+   functions are needed because they may lock FILE (so that we don't
+   read garbage when a concurrent process writes to FILE), but their
+   drawback is that they have a common global state.  */
+int read_utmp (char const *file, idx_t *n_entries, STRUCT_UTMP **utmp_buf,
+               int options);
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __READUTMP_H__ */
diff --git a/m4/gnulib-comp.m4 b/m4/gnulib-comp.m4
index 3382e9bc241..14ff92040a4 100644
--- a/m4/gnulib-comp.m4
+++ b/m4/gnulib-comp.m4
@@ -51,6 +51,7 @@ AC_DEFUN
   # Code from module at-internal:
   # Code from module attribute:
   # Code from module binary-io:
+  # Code from module boot-time:
   # Code from module builtin-expect:
   # Code from module byteswap:
   # Code from module c-ctype:
@@ -243,6 +244,7 @@ AC_DEFUN
   gl_ASSERT_H
   gl_CONDITIONAL_HEADER([assert.h])
   AC_PROG_MKDIR_P
+  gl_PREREQ_READUTMP_H
   gl___BUILTIN_EXPECT
   gl_BYTESWAP
   gl_CONDITIONAL_HEADER([byteswap.h])
@@ -1252,6 +1254,9 @@ AC_DEFUN
   lib/attribute.h
   lib/binary-io.c
   lib/binary-io.h
+  lib/boot-time-aux.h
+  lib/boot-time.c
+  lib/boot-time.h
   lib/byteswap.in.h
   lib/c++defs.h
   lib/c-ctype.c
@@ -1383,6 +1388,7 @@ AC_DEFUN
   lib/rawmemchr.valgrind
   lib/readlink.c
   lib/readlinkat.c
+  lib/readutmp.h
   lib/realloc.c
   lib/regcomp.c
   lib/regex.c
@@ -1542,6 +1548,7 @@ AC_DEFUN
   m4/rawmemchr.m4
   m4/readlink.m4
   m4/readlinkat.m4
+  m4/readutmp.m4
   m4/realloc.m4
   m4/regex.m4
   m4/sha1.m4
diff --git a/m4/readutmp.m4 b/m4/readutmp.m4
new file mode 100644
index 00000000000..fff8d4eb7bf
--- /dev/null
+++ b/m4/readutmp.m4
@@ -0,0 +1,117 @@
+# readutmp.m4 serial 28
+dnl Copyright (C) 2002-2023 Free Software Foundation, Inc.
+dnl This file is free software; the Free Software Foundation
+dnl gives unlimited permission to copy and/or distribute it,
+dnl with or without modifications, as long as this notice is preserved.
+
+AC_DEFUN([gl_READUTMP],
+[
+  AC_REQUIRE([gl_SYSTEMD_CHOICE])
+
+  dnl Set READUTMP_LIB to '-lsystemd' or '', depending on whether use of
+  dnl systemd APIs is possible and desired (only the systemd login API, here).
+  dnl AC_LIB_LINKFLAGS_BODY would be overkill here, since few people install
+  dnl libsystemd in non-system directories.
+  READUTMP_LIB=
+  if test "$SYSTEMD_CHOICE" = yes; then
+    AC_CHECK_HEADER([systemd/sd-login.h])
+    if test $ac_cv_header_systemd_sd_login_h = yes; then
+      AC_CACHE_CHECK([for libsystemd version >= 254],
+        [gl_cv_lib_readutmp_systemd],
+        [gl_save_LIBS="$LIBS"
+         LIBS="$LIBS -lsystemd"
+         AC_LINK_IFELSE(
+           [AC_LANG_PROGRAM([[
+              #include <stdint.h>
+              #include <systemd/sd-login.h>
+              ]], [[
+              uint64_t st;
+              sd_session_get_start_time ("1", &st);
+              ]])
+           ],
+           [gl_cv_lib_readutmp_systemd=yes],
+           [gl_cv_lib_readutmp_systemd=no])
+         LIBS="$gl_save_LIBS"
+        ])
+      if test $gl_cv_lib_readutmp_systemd = yes; then
+        AC_DEFINE([READUTMP_USE_SYSTEMD], [1],
+          [Define if the readutmp module should use the systemd login API.])
+        READUTMP_LIB='-lsystemd'
+      fi
+    fi
+  fi
+  AC_SUBST([READUTMP_LIB])
+
+  gl_PREREQ_READUTMP_H
+])
+
+# Prerequisites of readutmp.h and boot-time-aux.h.
+AC_DEFUN_ONCE([gl_PREREQ_READUTMP_H],
+[
+  dnl Persuade utmpx.h to declare utmpxname
+  AC_REQUIRE([AC_USE_SYSTEM_EXTENSIONS])
+
+  AC_CHECK_HEADERS_ONCE([utmp.h utmpx.h])
+  if test $ac_cv_header_utmp_h = yes || test $ac_cv_header_utmpx_h = yes; then
+    dnl Prerequisites of lib/readutmp.h and lib/readutmp.c.
+    AC_CHECK_FUNCS_ONCE([utmpname utmpxname])
+    AC_CHECK_DECLS([getutent],,,[[
+/* <sys/types.h> is a prerequisite of <utmp.h> on FreeBSD 8.0, OpenBSD 4.6.  */
+#include <sys/types.h>
+#ifdef HAVE_UTMP_H
+# include <utmp.h>
+#endif
+]])
+    utmp_includes="\
+AC_INCLUDES_DEFAULT
+#ifdef HAVE_UTMPX_H
+# include <utmpx.h>
+#endif
+#ifdef HAVE_UTMP_H
+# if defined _THREAD_SAFE && defined UTMP_DATA_INIT
+   /* When including both utmp.h and utmpx.h on AIX 4.3, with _THREAD_SAFE
+      defined, work around the duplicate struct utmp_data declaration.  */
+#  define utmp_data gl_aix_4_3_workaround_utmp_data
+# endif
+# include <utmp.h>
+#endif
+"
+    AC_CHECK_MEMBERS([struct utmpx.ut_user],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_user],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_name],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_name],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_type],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_type],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_pid],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_pid],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_tv],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_host],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_host],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_id],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_id],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_session],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_session],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_exit],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_exit],,,[$utmp_includes])
+
+    AC_CHECK_MEMBERS([struct utmpx.ut_exit.ut_exit],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_exit.e_exit],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_exit.e_exit],,,[$utmp_includes])
+
+    AC_CHECK_MEMBERS([struct utmpx.ut_exit.ut_termination],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmpx.ut_exit.e_termination],,,[$utmp_includes])
+    AC_CHECK_MEMBERS([struct utmp.ut_exit.e_termination],,,[$utmp_includes])
+  fi
+
+  AC_CHECK_HEADERS_ONCE([sys/param.h])
+  dnl <sys/sysctl.h> requires <sys/param.h> on OpenBSD 4.0.
+  AC_CHECK_HEADERS([sys/sysctl.h],,,
+    [AC_INCLUDES_DEFAULT
+     #if HAVE_SYS_PARAM_H
+     # include <sys/param.h>
+     #endif
+    ])
+  AC_CHECK_FUNCS([sysctl])
+
+  AC_CHECK_HEADERS_ONCE([OS.h])
+])
diff --git a/src/filelock.c b/src/filelock.c
index 3b1ff8ad566..d2161f1e58a 100644
--- a/src/filelock.c
+++ b/src/filelock.c
@@ -36,13 +36,9 @@ Copyright (C) 1985-1987, 1993-1994, 1996, 1998-2023 Free Software
 #include <sys/file.h>
 #include <fcntl.h>
 #include <unistd.h>
-
-#ifdef __FreeBSD__
-#include <sys/sysctl.h>
-#endif /* __FreeBSD__ */
-
 #include <errno.h>
 
+#include <boot-time.h>
 #include <c-ctype.h>
 
 #include "lisp.h"
@@ -55,20 +51,6 @@ Copyright (C) 1985-1987, 1993-1994, 1996, 1998-2023 Free Software
 
 #ifndef MSDOS
 
-#ifdef HAVE_UTMP_H
-#include <utmp.h>
-#endif
-
-/* Boot time is not available on Android.  */
-
-#if defined HAVE_ANDROID && !defined ANDROID_STUBIFY
-#undef BOOT_TIME
-#endif
-
-#if !defined WTMP_FILE && !defined WINDOWSNT && defined BOOT_TIME
-#define WTMP_FILE "/var/log/wtmp"
-#endif
-
 #ifdef HAVE_ANDROID
 #include "android.h" /* For `android_is_special_directory'.  */
 #endif /* HAVE_ANDROID */
@@ -127,153 +109,19 @@ #define WTMP_FILE "/var/log/wtmp"
 \f
 /* Return the time of the last system boot.  */
 
-static time_t boot_time;
-static bool boot_time_initialized;
-
-#ifdef BOOT_TIME
-static void get_boot_time_1 (const char *, bool);
-#endif
-
 static time_t
-get_boot_time (void)
+get_boot_sec (void)
 {
-  if (boot_time_initialized)
-    return boot_time;
-  boot_time_initialized = 1;
-
-#if defined (CTL_KERN) && defined (KERN_BOOTTIME)
-  {
-    int mib[2];
-    size_t size;
-    struct timeval boottime_val;
-
-    mib[0] = CTL_KERN;
-    mib[1] = KERN_BOOTTIME;
-    size = sizeof (boottime_val);
-
-    if (sysctl (mib, 2, &boottime_val, &size, NULL, 0) >= 0 && size != 0)
-      {
-	boot_time = boottime_val.tv_sec;
-	return boot_time;
-      }
-  }
-#endif /* defined (CTL_KERN) && defined (KERN_BOOTTIME) */
-
-#ifdef BOOT_TIME_FILE
-    {
-      struct stat st;
-      if (stat (BOOT_TIME_FILE, &st) == 0)
-	{
-	  boot_time = st.st_mtime;
-	  return boot_time;
-	}
-    }
-#endif /* BOOT_TIME_FILE */
-
-#if defined (BOOT_TIME)
-  /* The utmp routines maintain static state.  Don't touch that state
+  /* get_boot_time maintains static state.  Don't touch that state
      if we are going to dump, since it might not survive dumping.  */
   if (will_dump_p ())
-    return boot_time;
-
-  /* Try to get boot time from utmp before wtmp,
-     since utmp is typically much smaller than wtmp.
-     Passing a null pointer causes get_boot_time_1
-     to inspect the default file, namely utmp.  */
-  get_boot_time_1 (0, 0);
-  if (boot_time)
-    return boot_time;
-
-  /* Try to get boot time from the current wtmp file.  */
-  get_boot_time_1 (WTMP_FILE, 1);
-
-  /* If we did not find a boot time in wtmp, look at wtmp.1,
-     wtmp.1.gz, wtmp.2, wtmp.2.gz, and so on.  */
-  for (int counter = 0; counter < 20 && ! boot_time; counter++)
-    {
-      Lisp_Object filename = Qnil;
-      bool delete_flag = false;
-      char cmd_string[sizeof WTMP_FILE ".19.gz"];
-      AUTO_STRING_WITH_LEN (tempname, cmd_string,
-			    sprintf (cmd_string, "%s.%d", WTMP_FILE, counter));
-      if (! NILP (Ffile_exists_p (tempname)))
-	filename = tempname;
-      else
-	{
-	  tempname = make_formatted_string (cmd_string, "%s.%d.gz",
-					    WTMP_FILE, counter);
-	  if (! NILP (Ffile_exists_p (tempname)))
-	    {
-	      /* The utmp functions on older systems accept only file
-		 names up to 8 bytes long.  Choose a 2 byte prefix, so
-		 the 6-byte suffix does not make the name too long.  */
-	      filename = Fmake_temp_file_internal (build_string ("wt"), Qnil,
-						   empty_unibyte_string, Qnil);
-	      CALLN (Fcall_process, build_string ("gzip"), Qnil,
-		     list2 (QCfile, filename), Qnil,
-		     build_string ("-cd"), tempname);
-	      delete_flag = true;
-	    }
-	}
-
-      if (! NILP (filename))
-	{
-	  get_boot_time_1 (SSDATA (filename), 1);
-	  if (delete_flag)
-	    emacs_unlink (SSDATA (filename));
-	}
-    }
-
-  return boot_time;
-#else
-  return 0;
-#endif
-}
+    return 0;
 
-#ifdef BOOT_TIME
-/* Try to get the boot time from wtmp file FILENAME.
-   This succeeds if that file contains a reboot record.
-
-   If FILENAME is zero, use the same file as before;
-   if no FILENAME has ever been specified, this is the utmp file.
-   Use the newest reboot record if NEWEST,
-   the first reboot record otherwise.
-   Ignore all reboot records on or before BOOT_TIME.
-   Success is indicated by setting BOOT_TIME to a larger value.  */
-
-void
-get_boot_time_1 (const char *filename, bool newest)
-{
-  struct utmp ut, *utp;
-
-  if (filename)
-    utmpname (filename);
-
-  setutent ();
-
-  while (1)
-    {
-      /* Find the next reboot record.  */
-      ut.ut_type = BOOT_TIME;
-      utp = getutid (&ut);
-      if (! utp)
-	break;
-      /* Compare reboot times and use the newest one.  */
-      if (utp->ut_time > boot_time)
-	{
-	  boot_time = utp->ut_time;
-	  if (! newest)
-	    break;
-	}
-      /* Advance on element in the file
-	 so that getutid won't repeat the same one.  */
-      utp = getutent ();
-      if (! utp)
-	break;
-    }
-  endutent ();
+  struct timespec boot_time;
+  boot_time.tv_sec = 0;
+  get_boot_time (&boot_time);
+  return boot_time.tv_sec;
 }
-#endif /* BOOT_TIME */
 \f
 /* An arbitrary limit on lock contents length.  8 K should be plenty
    big enough in practice.  */
@@ -418,7 +266,7 @@ create_lock_file (char *lfname, char *lock_info_str, bool force)
 static int
 lock_file_1 (Lisp_Object lfname, bool force)
 {
-  intmax_t boot = get_boot_time ();
+  intmax_t boot = get_boot_sec ();
   Lisp_Object luser_name = Fuser_login_name (Qnil);
   Lisp_Object lhost_name = Fsystem_name ();
 
@@ -604,7 +452,7 @@ current_lock_owner (lock_info_type *owner, Lisp_Object lfname)
                && (kill (pid, 0) >= 0 || errno == EPERM)
 	       && (boot_time == 0
 		   || (boot_time <= TYPE_MAXIMUM (time_t)
-		       && within_one_second (boot_time, get_boot_time ()))))
+		       && within_one_second (boot_time, get_boot_sec ()))))
         return ANOTHER_OWNS_IT;
       /* The owner process is dead or has a strange pid, so try to
          zap the lockfile.  */
-- 
2.39.2


^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  2:49 ` boot-time: straighten code Paul Eggert
@ 2023-08-13  3:26   ` Po Lu
  2023-08-13  6:35     ` Paul Eggert
                       ` (3 more replies)
  2023-08-14  8:02   ` boot-time: straighten code Andreas Schwab
  2023-08-15 21:12   ` Bruno Haible
  2 siblings, 4 replies; 24+ messages in thread
From: Po Lu @ 2023-08-13  3:26 UTC (permalink / raw)
  To: Paul Eggert; +Cc: Bruno Haible, bug-gnulib, Emacs Development

Paul Eggert <eggert@cs.ucla.edu> writes:

> On 2023-08-11 14:49, Bruno Haible wrote:
>> Paul: With this simplification, you may consider using the 'boot-time' module
>> in Emacs. I bet that it produces a better result than Emacs' src/filelock.c
>> on many platforms. (I haven't tested it, but I could test it if you give me
>> a manual testing recipe.)
>
> Thanks for doing all that. I installed the attached patch into Emacs
> master, which you should be able to test via:
>
> 	git clone https://git.savannah.gnu.org/git/emacs.git
> 	cd emacs
> 	./autogen.sh
> 	./configure
> 	make
> 	src/emacs
>
> Please give it a try, especially on any MS-Windows platform you happen
> to have. I have tested only on Ubuntu 23.04 so far.
>
> A simple way to test is to use Emacs to start editing a file (without
> saving) and then inspect the symbolic link .#* that Emacs uses as a
> lock file. The trailing digits of that link's contents should be the
> boot time. These symlinks are Emacs's only use of boot time.

During the automated build of the Android port, the following errors
were encountered with the Android NDK r10b and __ANDROID_API__ set to 8
(which is the oldest configuration Emacs supports.)  Bruno, would you
please investigate this?

In file included from boot-time.c:54:0:
boot-time-aux.h: In function 'get_linux_uptime':
boot-time-aux.h:70:3: error: implicit declaration of function 'sysinfo' [-Werror=implicit-function-declaration]
   if (sysinfo (&info) >= 0)
   ^
boot-time.c: In function 'get_boot_time_uncached':
boot-time.c:111:26: error: 'BOOT_TIME' undeclared (first use in this function)
       if (ut->ut_type == BOOT_TIME)
                          ^
boot-time.c:111:26: note: each undeclared identifier is reported only once for each function it appears in
boot-time.c:126:3: error: implicit declaration of function 'endutent' [-Werror=implicit-function-declaration]
   END_UTMP_ENT ();
   ^
cc1: some warnings being treated as errors



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  3:26   ` Po Lu
@ 2023-08-13  6:35     ` Paul Eggert
  2023-08-13 13:45     ` Bruno Haible
                       ` (2 subsequent siblings)
  3 siblings, 0 replies; 24+ messages in thread
From: Paul Eggert @ 2023-08-13  6:35 UTC (permalink / raw)
  To: Po Lu; +Cc: Bruno Haible, bug-gnulib, Emacs Development

[-- Attachment #1: Type: text/plain, Size: 1224 bytes --]

On 2023-08-12 20:26, Po Lu wrote:

> In file included from boot-time.c:54:0:
> boot-time-aux.h: In function 'get_linux_uptime':
> boot-time-aux.h:70:3: error: implicit declaration of function 'sysinfo' [-Werror=implicit-function-declaration]
>     if (sysinfo (&info) >= 0)
>     ^
> boot-time.c: In function 'get_boot_time_uncached':
> boot-time.c:111:26: error: 'BOOT_TIME' undeclared (first use in this function)
>         if (ut->ut_type == BOOT_TIME)
>                            ^
> boot-time.c:111:26: note: each undeclared identifier is reported only once for each function it appears in
> boot-time.c:126:3: error: implicit declaration of function 'endutent' [-Werror=implicit-function-declaration]
>     END_UTMP_ENT ();
>     ^
> cc1: some warnings being treated as errors

Thanks for reporting that. As I understand it, the utmp/utmpx functions 
are a losing cause on Android since they never return anything. If so, 
it's simpler to bypass these functions on that platform. Also, Gnulib 
should bypass sysinfo unless it's available. Please try the attached 
patch, which I haven't installed onto Emacs master on Savannah. I've 
tested it only on Ubuntu 23.04.

If this patch works we can propagate it to Gnulib.

[-- Attachment #2: 0001-Temp-patch-for-Android.patch --]
[-- Type: text/x-patch, Size: 6598 bytes --]

From d2db6d8e92282de8ffb5293dd1445e3a2e549ed0 Mon Sep 17 00:00:00 2001
From: Paul Eggert <eggert@cs.ucla.edu>
Date: Sat, 12 Aug 2023 23:20:12 -0700
Subject: [PATCH] Temp patch for Android

---
 configure.ac        | 19 -------------------
 lib/boot-time-aux.h | 10 ++++++++++
 lib/boot-time.c     | 26 +++++++++-----------------
 src/conf_post.h     | 34 ----------------------------------
 src/filelock.c      | 14 +++++---------
 5 files changed, 24 insertions(+), 79 deletions(-)

diff --git a/configure.ac b/configure.ac
index 46836073aa0..0234a82b92f 100644
--- a/configure.ac
+++ b/configure.ac
@@ -2710,25 +2710,6 @@ AC_DEFUN
 
     # Check for some functions not always present in the NDK.
     AC_CHECK_DECLS([android_get_device_api_level])
-    AC_CHECK_DECLS([endutent, sysinfo], [], [],
-      [[
-#include <sys/sysinfo.h>
-#include <utmp.h>
-]])
-
-    # Establish if BOOT_TIME is defined in utmp.h.
-    AC_CACHE_CHECK([if utmp.h defines BOOT_TIME],
-      [emacs_cv_utmp_h_defines_boot_time],
-      [AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
-#include <utmp.h>
-#ifndef BOOT_TIME
-BOOT_TIME not defined
-#endif /* BOOT_TIME */
-]], [[]])], [emacs_cv_utmp_h_defines_boot_time=yes],
-      [emacs_cv_utmp_h_defines_boot_time=no])])
-    AS_IF([test x"$emacs_cv_utmp_h_defines_boot_time" = xyes],
-      [AC_DEFINE([UTMP_H_DEFINES_BOOT_TIME], [1],
-        [Define to 1 if building for Android and utmp.h declares BOOT_TIME])])
 
     # Say this build is really for Android.
     REALLY_ANDROID=yes])])
diff --git a/lib/boot-time-aux.h b/lib/boot-time-aux.h
index 348611fc85c..e782ca6eac6 100644
--- a/lib/boot-time-aux.h
+++ b/lib/boot-time-aux.h
@@ -16,6 +16,14 @@
 
 /* Written by Bruno Haible <bruno@clisp.org>.  */
 
+#if defined __linux__ || 9 <= __ANDROID_API__
+# include <sys/sysinfo.h>
+#endif
+#if 9 <= __ANDROID_API__
+/* Absent from some NDK versions, but present in API level 9+.  */
+extern int sysinfo (struct sysinfo *);
+#endif
+
 #define SIZEOF(a) (sizeof(a)/sizeof(a[0]))
 
 #if defined __linux__ || defined __ANDROID__
@@ -65,6 +73,7 @@ get_linux_uptime (struct timespec *p_uptime)
     }
 # endif
 
+# if defined __linux__ || 9 <= __ANDROID_API__
   /* The sysinfo call returns the uptime with a resolution of 1 sec only.  */
   struct sysinfo info;
   if (sysinfo (&info) >= 0)
@@ -73,6 +82,7 @@ get_linux_uptime (struct timespec *p_uptime)
       p_uptime->tv_nsec = 0;
       return 0;
     }
+# endif
 
   return -1;
 }
diff --git a/lib/boot-time.c b/lib/boot-time.c
index d813bfa5825..331711238bc 100644
--- a/lib/boot-time.c
+++ b/lib/boot-time.c
@@ -27,11 +27,6 @@
 #include <sys/types.h>
 #include <sys/stat.h>
 
-#if defined __linux__ || defined __ANDROID__
-# include <sys/sysinfo.h>
-# include <time.h>
-#endif
-
 #if HAVE_SYS_SYSCTL_H && !defined __minix
 # if HAVE_SYS_PARAM_H
 #  include <sys/param.h>
@@ -76,7 +71,12 @@ get_boot_time_uncached (struct timespec *p_boot_time)
 {
   struct timespec found_boot_time = {0};
 
-# if (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_TYPE : HAVE_STRUCT_UTMP_UT_TYPE)
+# ifdef __ANDROID__
+
+  /* Workaround for Android:  */
+  get_android_boot_time (&found_boot_time);
+
+# elif (HAVE_UTMPX_H ? HAVE_STRUCT_UTMPX_UT_TYPE : HAVE_STRUCT_UTMP_UT_TYPE)
 
   /* Try to find the boot time in the /var/run/utmp file.  */
 
@@ -90,7 +90,7 @@ get_boot_time_uncached (struct timespec *p_boot_time)
 
   SET_UTMP_ENT ();
 
-#   if (defined __linux__ && !defined __ANDROID__) || defined __minix
+#   if defined __linux__ || defined __minix
   /* Timestamp of the "runlevel" entry, if any.  */
   struct timespec runlevel_ts = {0};
 #   endif
@@ -111,7 +111,7 @@ get_boot_time_uncached (struct timespec *p_boot_time)
       if (ut->ut_type == BOOT_TIME)
         found_boot_time = ts;
 
-#   if defined __linux__ && !defined __ANDROID__
+#   ifdef __linux__
       if (memcmp (UT_USER (ut), "runlevel", strlen ("runlevel") + 1) == 0
           && memcmp (ut->ut_line, "~", strlen ("~") + 1) == 0)
         runlevel_ts = ts;
@@ -125,7 +125,7 @@ get_boot_time_uncached (struct timespec *p_boot_time)
 
   END_UTMP_ENT ();
 
-#   if defined __linux__ && !defined __ANDROID__
+#   ifdef __linux__
   /* On Raspbian, which runs on hardware without a real-time clock, during boot,
        1. the clock gets set to 1970-01-01 00:00:00,
        2. an entry gets written into /var/run/utmp, with ut_type = BOOT_TIME,
@@ -145,14 +145,6 @@ get_boot_time_uncached (struct timespec *p_boot_time)
     }
 #   endif
 
-#   if defined __ANDROID__
-  if (found_boot_time.tv_sec == 0)
-    {
-      /* Workaround for Android:  */
-      get_android_boot_time (&found_boot_time);
-    }
-#   endif
-
 #   if defined __minix
   /* On Minix, during boot,
        1. an entry gets written into /var/run/utmp, with ut_type = BOOT_TIME,
diff --git a/src/conf_post.h b/src/conf_post.h
index 5f18e5ae4bb..f31e012dc6e 100644
--- a/src/conf_post.h
+++ b/src/conf_post.h
@@ -471,37 +471,3 @@ #define VFORK() vfork ()
 #undef MB_CUR_MAX
 #define MB_CUR_MAX REPLACEMENT_MB_CUR_MAX
 #endif /* REPLACEMENT_MB_CUR_MAX */
-
-#ifdef __ANDROID__
-
-/* The Android NDK r10b omits the function `endutent' that is actually
-   defined in the C library and used by Gnulib.  Define a prototype
-   for it here.  */
-
-#if !HAVE_DECL_ENDUTENT
-extern void endutent (void);
-#endif /* !HAVE_DECL_ENDUTENT */
-
-/* Now define substitutes for BOOT_TIME if necessary.  */
-
-#ifndef UTMP_H_DEFINES_BOOT_TIME
-#include <utmp.h>
-
-#define BOOT_TIME 2
-#endif /* !UTMP_H_DEFINES_BOOT_TIME */
-
-/* sysinfo is also absent from some versions of the NDK, yet is
-   present on API level 9 and above.  */
-
-#if !HAVE_DECL_SYSINFO
-#include <sys/sysinfo.h>
-
-#if __ANDROID_API__ >= 9
-extern int sysinfo (struct sysinfo *info);
-#else /* __ANDROID_API__ < 8 */
-/* Gnulib uses this function unconditionally.  */
-#define sysinfo(ignored) ((void) ignored, (errno = ENOSYS), -1)
-#endif /* __ANDROID_API >= 9 */
-#endif /* !HAVE_DECL_SYSINFO */
-
-#endif /* __ANDROID__ */
diff --git a/src/filelock.c b/src/filelock.c
index f3075b93322..92be20ad8a0 100644
--- a/src/filelock.c
+++ b/src/filelock.c
@@ -118,16 +118,12 @@ get_boot_sec (void)
   if (will_dump_p ())
     return 0;
 
+  struct timespec boot_time;
+  boot_time.tv_sec = 0;
 #ifndef MSDOS
-  {
-    struct timespec boot_time;
-    boot_time.tv_sec = 0;
-    get_boot_time (&boot_time);
-    return boot_time.tv_sec;
-  }
-#else /* MSDOS */
-  return 0;
-#endif /* MSDOS */
+  get_boot_time (&boot_time);
+#endif
+  return boot_time.tv_sec;
 }
 \f
 /* An arbitrary limit on lock contents length.  8 K should be plenty
-- 
2.39.2


^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  3:26   ` Po Lu
  2023-08-13  6:35     ` Paul Eggert
@ 2023-08-13 13:45     ` Bruno Haible
  2023-08-13 14:16     ` Bruno Haible
  2023-08-13 14:36     ` Bruno Haible
  3 siblings, 0 replies; 24+ messages in thread
From: Bruno Haible @ 2023-08-13 13:45 UTC (permalink / raw)
  To: Paul Eggert, Po Lu; +Cc: bug-gnulib, Emacs-devel

Po Lu wrote:
> During the automated build of the Android port, the following errors
> were encountered with the Android NDK r10b and __ANDROID_API__ set to 8
> (which is the oldest configuration Emacs supports.)
> ...
> boot-time.c:111:26: note: each undeclared identifier is reported only once for each function it appears in
> boot-time.c:126:3: error: implicit declaration of function 'endutent' [-Werror=implicit-function-declaration]
>    END_UTMP_ENT ();
>    ^

endutent() exists in Android, already in API 8. It's just that its declaration
is missing from <utmp.h> until ca. 2015.

Should be fixed through this patch. Also, it removes a declaration that was
needed only for Ultrix 4.3. Gnulib dropped Ultrix support in 2018.

Paul Eggert wrote:
> As I understand it, the utmp/utmpx functions 
> are a losing cause on Android since they never return anything. If so, 
> it's simpler to bypass these functions on that platform.

It's more future-proof to continue to use the API that they have. Who knows,
they might add useful data into the UTMP_FILE at some point. (E.g. they had a
broken localeconv() function for a long time; then they actually fixed it.)


2023-08-13  Bruno Haible  <bruno@clisp.org>

	readutmp, boot-time: Fix compilation error on old Android.
	Reported by Po Lu in
	<https://lists.gnu.org/archive/html/bug-gnulib/2023-08/msg00108.html>.
	* lib/readutmp.c (endutent): New fallback declaration, for Android.
	(getutent): Remove Ultrix workaround from 2000-04-05.
	* lib/boot-time.c: Likewise.
	* m4/readutmp.m4 (gl_PREREQ_READUTMP_H): Test whether endutent is
	declared, not getutent.
	* doc/glibc-functions/endutent.texi: Mention the Android bug.

diff --git a/doc/glibc-functions/endutent.texi b/doc/glibc-functions/endutent.texi
index 5e12e23c44..b4f3dd397a 100644
--- a/doc/glibc-functions/endutent.texi
+++ b/doc/glibc-functions/endutent.texi
@@ -28,4 +28,7 @@
 @item
 This function is missing on some platforms:
 macOS 11.1, FreeBSD 13.0, OpenBSD 6.7, Minix 3.1.8, mingw, MSVC 14.
+@item
+This function is not declared on some platforms:
+Android before ca.@: 2015.
 @end itemize
diff --git a/lib/boot-time.c b/lib/boot-time.c
index d813bfa582..c359954f19 100644
--- a/lib/boot-time.c
+++ b/lib/boot-time.c
@@ -65,8 +65,10 @@
 # define UT_USER(UT) ((UT)->ut_user)
 #endif
 
-#if !HAVE_UTMPX_H && HAVE_UTMP_H && defined UTMP_NAME_FUNCTION && !HAVE_DECL_GETUTENT
-struct utmp *getutent (void);
+#if !HAVE_UTMPX_H && HAVE_UTMP_H && defined UTMP_NAME_FUNCTION
+# if !HAVE_DECL_ENDUTENT /* Android */
+void endutent (void);
+# endif
 #endif
 
 #if defined __linux__ || HAVE_UTMPX_H || HAVE_UTMP_H || defined __CYGWIN__ || defined _WIN32
diff --git a/lib/readutmp.c b/lib/readutmp.c
index ef9f0aff43..0b7732b165 100644
--- a/lib/readutmp.c
+++ b/lib/readutmp.c
@@ -314,9 +314,11 @@ have_boot_time (struct utmp_alloc a)
   return false;
 }
 
-# if !HAVE_UTMPX_H && HAVE_UTMP_H && defined UTMP_NAME_FUNCTION && !HAVE_DECL_GETUTENT
-struct utmp *getutent (void);
+#if !HAVE_UTMPX_H && HAVE_UTMP_H && defined UTMP_NAME_FUNCTION
+# if !HAVE_DECL_ENDUTENT /* Android */
+void endutent (void);
 # endif
+#endif
 
 static int
 read_utmp_from_file (char const *file, idx_t *n_entries, STRUCT_UTMP **utmp_buf,
diff --git a/m4/readutmp.m4 b/m4/readutmp.m4
index fff8d4eb7b..9dffe981b8 100644
--- a/m4/readutmp.m4
+++ b/m4/readutmp.m4
@@ -1,4 +1,4 @@
-# readutmp.m4 serial 28
+# readutmp.m4 serial 29
 dnl Copyright (C) 2002-2023 Free Software Foundation, Inc.
 dnl This file is free software; the Free Software Foundation
 dnl gives unlimited permission to copy and/or distribute it,
@@ -55,7 +55,7 @@ AC_DEFUN_ONCE([gl_PREREQ_READUTMP_H]
   if test $ac_cv_header_utmp_h = yes || test $ac_cv_header_utmpx_h = yes; then
     dnl Prerequisites of lib/readutmp.h and lib/readutmp.c.
     AC_CHECK_FUNCS_ONCE([utmpname utmpxname])
-    AC_CHECK_DECLS([getutent],,,[[
+    AC_CHECK_DECLS([endutent],,,[[
 /* <sys/types.h> is a prerequisite of <utmp.h> on FreeBSD 8.0, OpenBSD 4.6.  */
 #include <sys/types.h>
 #ifdef HAVE_UTMP_H






^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  3:26   ` Po Lu
  2023-08-13  6:35     ` Paul Eggert
  2023-08-13 13:45     ` Bruno Haible
@ 2023-08-13 14:16     ` Bruno Haible
  2023-08-13 14:36     ` Bruno Haible
  3 siblings, 0 replies; 24+ messages in thread
From: Bruno Haible @ 2023-08-13 14:16 UTC (permalink / raw)
  To: Paul Eggert, Po Lu; +Cc: bug-gnulib, Emacs-devel

Po Lu wrote:
> During the automated build of the Android port, the following errors
> were encountered with the Android NDK r10b and __ANDROID_API__ set to 8
> (which is the oldest configuration Emacs supports.)...
> 
> In file included from boot-time.c:54:0:
> boot-time-aux.h: In function 'get_linux_uptime':
> boot-time-aux.h:70:3: error: implicit declaration of function 'sysinfo' [-Werror=implicit-function-declaration]
>    if (sysinfo (&info) >= 0)
>    ^

Should be fixed by the patch below.

Paul Eggert wrote:
> Gnulib should bypass sysinfo unless it's available.

Yes. Note that it's available in Android libc already at API level 3. It's only
the declaration that was added in API level 9.

Also, if we write '#if HAVE_DECL_SYSINFO' instead of '9 <= __ANDROID_API__',
the change might be helpful also with other Linux-based libcs that may be
developed in the future.


2023-08-13  Bruno Haible  <bruno@clisp.org>

	readutmp, boot-time: Fix compilation error on Android API 8.
	Reported by Po Lu in
	<https://lists.gnu.org/archive/html/bug-gnulib/2023-08/msg00108.html>.
	* m4/readutmp.m4 (gl_PREREQ_READUTMP_H): Test whether sysinfo is
	declared.
	* lib/boot-time-aux.h (get_linux_uptime): Invoke sysinfo only if it is
	declared.
	* doc/glibc-functions/sysinfo.texi: Mention the Android problem.

diff --git a/doc/glibc-functions/sysinfo.texi b/doc/glibc-functions/sysinfo.texi
index 3b1ade31a6..16e12b12ac 100644
--- a/doc/glibc-functions/sysinfo.texi
+++ b/doc/glibc-functions/sysinfo.texi
@@ -17,4 +17,7 @@
 @item
 This function is missing on some platforms:
 macOS 11.1, FreeBSD 13.0, NetBSD 9.0, OpenBSD 6.7, Minix 3.1.8, AIX 5.1, Cygwin 1.7.9, mingw, MSVC 14.
+@item
+This function is not declared and thus not part of the Android API
+for Android API levels < 9.
 @end itemize
diff --git a/lib/boot-time-aux.h b/lib/boot-time-aux.h
index 348611fc85..e59a0fd03c 100644
--- a/lib/boot-time-aux.h
+++ b/lib/boot-time-aux.h
@@ -65,6 +65,7 @@ get_linux_uptime (struct timespec *p_uptime)
     }
 # endif
 
+# if HAVE_DECL_SYSINFO /* not available in Android API < 9 */
   /* The sysinfo call returns the uptime with a resolution of 1 sec only.  */
   struct sysinfo info;
   if (sysinfo (&info) >= 0)
@@ -73,6 +74,7 @@ get_linux_uptime (struct timespec *p_uptime)
       p_uptime->tv_nsec = 0;
       return 0;
     }
+# endif
 
   return -1;
 }
diff --git a/m4/readutmp.m4 b/m4/readutmp.m4
index 9dffe981b8..0a47f4bb77 100644
--- a/m4/readutmp.m4
+++ b/m4/readutmp.m4
@@ -1,4 +1,4 @@
-# readutmp.m4 serial 29
+# readutmp.m4 serial 30
 dnl Copyright (C) 2002-2023 Free Software Foundation, Inc.
 dnl This file is free software; the Free Software Foundation
 dnl gives unlimited permission to copy and/or distribute it,
@@ -103,6 +103,10 @@ AC_DEFUN_ONCE([gl_PREREQ_READUTMP_H]
     AC_CHECK_MEMBERS([struct utmp.ut_exit.e_termination],,,[$utmp_includes])
   fi
 
+  AC_CHECK_DECLS([sysinfo],,,[[
+    #include <sys/sysinfo.h>
+    ]])
+
   AC_CHECK_HEADERS_ONCE([sys/param.h])
   dnl <sys/sysctl.h> requires <sys/param.h> on OpenBSD 4.0.
   AC_CHECK_HEADERS([sys/sysctl.h],,,






^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  3:26   ` Po Lu
                       ` (2 preceding siblings ...)
  2023-08-13 14:16     ` Bruno Haible
@ 2023-08-13 14:36     ` Bruno Haible
  2023-08-13 23:44       ` Po Lu
  3 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-13 14:36 UTC (permalink / raw)
  To: Paul Eggert, Po Lu; +Cc: bug-gnulib, Emacs-devel

Po Lu wrote:
> During the automated build of the Android port, the following errors
> were encountered with the Android NDK r10b and __ANDROID_API__ set to 8
> (which is the oldest configuration Emacs supports.)
> ...
> boot-time.c: In function 'get_boot_time_uncached':
> boot-time.c:111:26: error: 'BOOT_TIME' undeclared (first use in this function)
>        if (ut->ut_type == BOOT_TIME)
>                           ^

Unlike USER_PROCESS, which is defined in all versions of Android's <utmp.h>,
BOOT_TIME is only defined in newer versions.

When compiling against an older version, we need to use the value from the
newer versions — otherwise a binary built against an older version might
not work right when running on a newer Android.

This patch does it.


2023-08-13  Bruno Haible  <bruno@clisp.org>

	readutmp, boot-time: Fix compilation error on old Android.
	Reported by Po Lu in
	<https://lists.gnu.org/archive/html/bug-gnulib/2023-08/msg00108.html>.
	* lib/readutmp.h (BOOT_TIME): Add fallback.

diff --git a/lib/readutmp.h b/lib/readutmp.h
index 1cf588d265..f7cad36d44 100644
--- a/lib/readutmp.h
+++ b/lib/readutmp.h
@@ -249,6 +249,13 @@ struct utmpx32
 # define WTMP_FILE "/etc/wtmp"
 #endif
 
+/* In early versions of Android, <utmp.h> did not define BOOT_TIME, only
+   USER_PROCESS.  We need to use the value that is defined in newer versions
+   of Android.  */
+#if defined __ANDROID__ && !defined BOOT_TIME
+# define BOOT_TIME 2
+#endif
+
 /* Some platforms, such as OpenBSD, don't have an ut_type field and don't have
    the BOOT_TIME and USER_PROCESS macros.  But we want to support them in
    'struct gl_utmp'.  */






^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13 14:36     ` Bruno Haible
@ 2023-08-13 23:44       ` Po Lu
  2023-08-13 23:59         ` Bruno Haible
  0 siblings, 1 reply; 24+ messages in thread
From: Po Lu @ 2023-08-13 23:44 UTC (permalink / raw)
  To: Bruno Haible; +Cc: Paul Eggert, bug-gnulib, Emacs-devel

Bruno Haible <bruno@clisp.org> writes:

> Po Lu wrote:
>> During the automated build of the Android port, the following errors
>> were encountered with the Android NDK r10b and __ANDROID_API__ set to 8
>> (which is the oldest configuration Emacs supports.)
>> ...
>> boot-time.c: In function 'get_boot_time_uncached':
>> boot-time.c:111:26: error: 'BOOT_TIME' undeclared (first use in this function)
>>        if (ut->ut_type == BOOT_TIME)
>>                           ^
>
> Unlike USER_PROCESS, which is defined in all versions of Android's <utmp.h>,
> BOOT_TIME is only defined in newer versions.
>
> When compiling against an older version, we need to use the value from the
> newer versions — otherwise a binary built against an older version might
> not work right when running on a newer Android.
>
> This patch does it.
>
>
> 2023-08-13  Bruno Haible  <bruno@clisp.org>
>
> 	readutmp, boot-time: Fix compilation error on old Android.
> 	Reported by Po Lu in
> 	<https://lists.gnu.org/archive/html/bug-gnulib/2023-08/msg00108.html>.
> 	* lib/readutmp.h (BOOT_TIME): Add fallback.
>
> diff --git a/lib/readutmp.h b/lib/readutmp.h
> index 1cf588d265..f7cad36d44 100644
> --- a/lib/readutmp.h
> +++ b/lib/readutmp.h
> @@ -249,6 +249,13 @@ struct utmpx32
>  # define WTMP_FILE "/etc/wtmp"
>  #endif
>  
> +/* In early versions of Android, <utmp.h> did not define BOOT_TIME, only
> +   USER_PROCESS.  We need to use the value that is defined in newer versions
> +   of Android.  */
> +#if defined __ANDROID__ && !defined BOOT_TIME
> +# define BOOT_TIME 2
> +#endif
> +
>  /* Some platforms, such as OpenBSD, don't have an ut_type field and don't have
>     the BOOT_TIME and USER_PROCESS macros.  But we want to support them in
>     'struct gl_utmp'.  */

Both of your patches appear to work.
Thanks.



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13 23:44       ` Po Lu
@ 2023-08-13 23:59         ` Bruno Haible
  2023-08-14  1:07           ` Po Lu
  0 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-13 23:59 UTC (permalink / raw)
  To: Po Lu; +Cc: Paul Eggert, bug-gnulib, Emacs-devel

Po Lu wrote:
> Both of your patches appear to work.

Thanks for the confirmation. As you know, I'm using a Termux environment
for testing on Android. This has only approximately, not exactly, the same
include files and libc as what you are using when you compile with some
Android NDK. Therefore I'm relying on your reports for the (few) portability
problems that arise because of differences between Android API levels
or between Android versions.

Bruno






^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13 23:59         ` Bruno Haible
@ 2023-08-14  1:07           ` Po Lu
  2023-08-14  2:14             ` Corwin Brust
  0 siblings, 1 reply; 24+ messages in thread
From: Po Lu @ 2023-08-14  1:07 UTC (permalink / raw)
  To: Bruno Haible; +Cc: Paul Eggert, bug-gnulib, Emacs-devel

Bruno Haible <bruno@clisp.org> writes:

> Po Lu wrote:
>> Both of your patches appear to work.
>
> Thanks for the confirmation. As you know, I'm using a Termux environment
> for testing on Android. This has only approximately, not exactly, the same
> include files and libc as what you are using when you compile with some
> Android NDK. Therefore I'm relying on your reports for the (few) portability
> problems that arise because of differences between Android API levels
> or between Android versions.
>
> Bruno

Regular merges from Gnulib into Emacs should facilitate that, since I've
set aside a machine for the express purpose of building the Android port
automatically in response to every large change to the branch.

And thanks again for your help.



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14  1:07           ` Po Lu
@ 2023-08-14  2:14             ` Corwin Brust
  2023-08-15 19:57               ` Windows port binaries Bruno Haible
  0 siblings, 1 reply; 24+ messages in thread
From: Corwin Brust @ 2023-08-14  2:14 UTC (permalink / raw)
  To: Po Lu; +Cc: Bruno Haible, Paul Eggert, bug-gnulib, Emacs-devel

On Sun, Aug 13, 2023 at 8:07 PM Po Lu <luangruo@yahoo.com> wrote:
>
> Bruno Haible <bruno@clisp.org> writes:
>
> > Po Lu wrote:
> >> Both of your patches appear to work.
> >
> > Thanks for the confirmation. As you know, I'm using a Termux environment
>
> Regular merges from Gnulib into Emacs should facilitate that, since I've
> set aside a machine for the express purpose of building the Android port
> automatically in response to every large change to the branch.
>

In case it helps others to assist with testing, for the moment I am
aggressively rebuilding the Windows port also, from (not quite every)
commit to emacs-29 or master, posting to:

  https://corwin.bru.st/emacs-29
  https://corwin.bru.st/emacs-30

Note, older release sets get all but the no-deps and source archives
removed; unpack the no-deps archive atop an existing install of the
same version of Emacs to use them for testing purposes after the
installer and full-zip have been pruned.



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  2:49 ` boot-time: straighten code Paul Eggert
  2023-08-13  3:26   ` Po Lu
@ 2023-08-14  8:02   ` Andreas Schwab
  2023-08-14  9:15     ` Bruno Haible
  2023-08-15 21:12   ` Bruno Haible
  2 siblings, 1 reply; 24+ messages in thread
From: Andreas Schwab @ 2023-08-14  8:02 UTC (permalink / raw)
  To: Paul Eggert; +Cc: Bruno Haible, bug-gnulib, Emacs Development

In file included from boot-time.c:47:
readutmp.h:145:16: error: ‘__UT_USERSIZE’ undeclared here (not in a function); did you mean ‘UT_USER_SIZE’?
  145 |   char ut_user[__UT_USERSIZE];     /* Username.  */
      |                ^~~~~~~~~~~~~
      |                UT_USER_SIZE
make[3]: *** [Makefile:102: boot-time.o] Error 1

-- 
Andreas Schwab, SUSE Labs, schwab@suse.de
GPG Key fingerprint = 0196 BAD8 1CE9 1970 F4BE  1748 E4D4 88E3 0EEA B9D7
"And now for something completely different."



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14  8:02   ` boot-time: straighten code Andreas Schwab
@ 2023-08-14  9:15     ` Bruno Haible
  2023-08-14  9:20       ` Andreas Schwab
  0 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-14  9:15 UTC (permalink / raw)
  To: Paul Eggert, Andreas Schwab; +Cc: bug-gnulib, Emacs Development

Andreas Schwab wrote:
> In file included from boot-time.c:47:
> readutmp.h:145:16: error: ‘__UT_USERSIZE’ undeclared here (not in a function); did you mean ‘UT_USER_SIZE’?
>   145 |   char ut_user[__UT_USERSIZE];     /* Username.  */
>       |                ^~~~~~~~~~~~~
>       |                UT_USER_SIZE
> make[3]: *** [Makefile:102: boot-time.o] Error 1

On which distro or glibc version, please?

Bruno






^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14  9:15     ` Bruno Haible
@ 2023-08-14  9:20       ` Andreas Schwab
  2023-08-14 10:19         ` Bruno Haible
  0 siblings, 1 reply; 24+ messages in thread
From: Andreas Schwab @ 2023-08-14  9:20 UTC (permalink / raw)
  To: Bruno Haible; +Cc: Paul Eggert, bug-gnulib, Emacs Development

On Aug 14 2023, Bruno Haible wrote:

> Andreas Schwab wrote:
>> In file included from boot-time.c:47:
>> readutmp.h:145:16: error: ‘__UT_USERSIZE’ undeclared here (not in a function); did you mean ‘UT_USER_SIZE’?
>>   145 |   char ut_user[__UT_USERSIZE];     /* Username.  */
>>       |                ^~~~~~~~~~~~~
>>       |                UT_USER_SIZE
>> make[3]: *** [Makefile:102: boot-time.o] Error 1
>
> On which distro or glibc version, please?

Any.

-- 
Andreas Schwab, SUSE Labs, schwab@suse.de
GPG Key fingerprint = 0196 BAD8 1CE9 1970 F4BE  1748 E4D4 88E3 0EEA B9D7
"And now for something completely different."



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14  9:20       ` Andreas Schwab
@ 2023-08-14 10:19         ` Bruno Haible
  2023-08-14 10:33           ` Andreas Schwab
  0 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-14 10:19 UTC (permalink / raw)
  To: Andreas Schwab; +Cc: Paul Eggert, bug-gnulib, Emacs-devel

Andreas Schwab wrote:
> >> In file included from boot-time.c:47:
> >> readutmp.h:145:16: error: ‘__UT_USERSIZE’ undeclared here (not in a function); did you mean ‘UT_USER_SIZE’?
> >>   145 |   char ut_user[__UT_USERSIZE];     /* Username.  */
> >>       |                ^~~~~~~~~~~~~
> >>       |                UT_USER_SIZE
> >> make[3]: *** [Makefile:102: boot-time.o] Error 1
> >
> > On which distro or glibc version, please?
> 
> Any.

Ah, you mean: on any 32-bit build with glibc.

Fixed through this patch:


2023-08-14  Bruno Haible  <bruno@clisp.org>

	readutmp, boot-time: Fix build on 32-bit glibc (regression 2023-08-11).
	Reported by Andreas Schwab <schwab@suse.de> in
	<https://lists.gnu.org/archive/html/bug-gnulib/2023-08/msg00125.html>.
	* lib/readutmp.h (struct utmpx32): Reference __UT_NAMESIZE, not
	__UT_USERSIZE.

diff --git a/lib/readutmp.h b/lib/readutmp.h
index f7cad36d44..1fbe29d86f 100644
--- a/lib/readutmp.h
+++ b/lib/readutmp.h
@@ -142,7 +142,7 @@ struct utmpx32
   pid_t ut_pid;                    /* Process ID of login process.  */
   char ut_line[__UT_LINESIZE];     /* Devicename.  */
   char ut_id[4];                   /* Inittab ID.  */
-  char ut_user[__UT_USERSIZE];     /* Username.  */
+  char ut_user[__UT_NAMESIZE];     /* Username.  */
   char ut_host[__UT_HOSTSIZE];     /* Hostname for remote login. */
   struct __exit_status ut_exit;    /* Exit status of a process marked
                                       as DEAD_PROCESS.  */






^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14 10:19         ` Bruno Haible
@ 2023-08-14 10:33           ` Andreas Schwab
  2023-08-14 13:51             ` Bruno Haible
  0 siblings, 1 reply; 24+ messages in thread
From: Andreas Schwab @ 2023-08-14 10:33 UTC (permalink / raw)
  To: Bruno Haible; +Cc: Paul Eggert, bug-gnulib, Emacs-devel

On Aug 14 2023, Bruno Haible wrote:

> diff --git a/lib/readutmp.h b/lib/readutmp.h
> index f7cad36d44..1fbe29d86f 100644
> --- a/lib/readutmp.h
> +++ b/lib/readutmp.h
> @@ -142,7 +142,7 @@ struct utmpx32
>    pid_t ut_pid;                    /* Process ID of login process.  */
>    char ut_line[__UT_LINESIZE];     /* Devicename.  */
>    char ut_id[4];                   /* Inittab ID.  */
> -  char ut_user[__UT_USERSIZE];     /* Username.  */
> +  char ut_user[__UT_NAMESIZE];     /* Username.  */
>    char ut_host[__UT_HOSTSIZE];     /* Hostname for remote login. */

That's still using reserved symbols.

-- 
Andreas Schwab, SUSE Labs, schwab@suse.de
GPG Key fingerprint = 0196 BAD8 1CE9 1970 F4BE  1748 E4D4 88E3 0EEA B9D7
"And now for something completely different."



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14 10:33           ` Andreas Schwab
@ 2023-08-14 13:51             ` Bruno Haible
  2023-08-15 23:03               ` Paul Eggert
  0 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-14 13:51 UTC (permalink / raw)
  To: Andreas Schwab; +Cc: Paul Eggert, bug-gnulib, Emacs-devel

Andreas Schwab wrote:
> >    char ut_host[__UT_HOSTSIZE];     /* Hostname for remote login. */
> 
> That's still using reserved symbols.

Fixed through the patch below.

But that is just a workaround. Is someone among the glibc people looking at the
original glibc bug https://sourceware.org/bugzilla/show_bug.cgi?id=30701 ?


2023-08-14  Bruno Haible  <bruno@clisp.org>

	readutmp, boot-time: Don't use __UT_* symbols (regression 2023-08-11).
	* lib/readutmp.h (_GL_UT_USER_SIZE, _GL_UT_ID_SIZE, _GL_UT_LINE_SIZE,
	_GL_UT_HOST_SIZE): New macros.
	(struct utmpx32): Use them.

diff --git a/lib/readutmp.h b/lib/readutmp.h
index 1fbe29d86f..3ddecf3727 100644
--- a/lib/readutmp.h
+++ b/lib/readutmp.h
@@ -136,14 +136,18 @@ enum { UT_HOST_SIZE = -1 };
 /* This is a near-copy of glibc's struct utmpx, which stops working
    after the year 2038.  Unlike the glibc version, struct utmpx32
    describes the file format even if time_t is 64 bits.  */
+#define _GL_UT_USER_SIZE  sizeof (((struct utmpx *) 0)->ut_user)
+#define _GL_UT_ID_SIZE    sizeof (((struct utmpx *) 0)->ut_id)
+#define _GL_UT_LINE_SIZE  sizeof (((struct utmpx *) 0)->ut_line)
+#define _GL_UT_HOST_SIZE  sizeof (((struct utmpx *) 0)->ut_host)
 struct utmpx32
 {
   short int ut_type;               /* Type of login.  */
   pid_t ut_pid;                    /* Process ID of login process.  */
-  char ut_line[__UT_LINESIZE];     /* Devicename.  */
-  char ut_id[4];                   /* Inittab ID.  */
-  char ut_user[__UT_NAMESIZE];     /* Username.  */
-  char ut_host[__UT_HOSTSIZE];     /* Hostname for remote login. */
+  char ut_line[_GL_UT_LINE_SIZE];  /* Devicename.  */
+  char ut_id[_GL_UT_ID_SIZE];      /* Inittab ID.  */
+  char ut_user[_GL_UT_USER_SIZE];  /* Username.  */
+  char ut_host[_GL_UT_HOST_SIZE];  /* Hostname for remote login. */
   struct __exit_status ut_exit;    /* Exit status of a process marked
                                       as DEAD_PROCESS.  */
   /* The fields ut_session and ut_tv must be the same size when compiled






^ permalink raw reply related	[flat|nested] 24+ messages in thread

* Re: Windows port binaries
  2023-08-14  2:14             ` Corwin Brust
@ 2023-08-15 19:57               ` Bruno Haible
  2023-08-16  6:45                 ` Po Lu
  2023-08-16 11:36                 ` Eli Zaretskii
  0 siblings, 2 replies; 24+ messages in thread
From: Bruno Haible @ 2023-08-15 19:57 UTC (permalink / raw)
  To: Corwin Brust, Emacs-devel; +Cc: Po Lu, Paul Eggert

Corwin Brust wrote:
> In case it helps others to assist with testing, for the moment I am
> aggressively rebuilding the Windows port also, from (not quite every)
> commit to emacs-29 or master, posting to:
> 
>   https://corwin.bru.st/emacs-29
>   https://corwin.bru.st/emacs-30

Thanks. I tried to use the
https://corwin.bru.st/emacs-30/emacs-30-latest-no-deps.zip
binary from today, but they don't work for me (on Windows 10), because
they rely on two DLLs which are not contained in the 'bin' directory:
  - libgmp-10.dll
  - libwinpthread-1.dll

Find below the output of "dumpbin.exe /imports emacs.exe".

Additionally, I find it strange:

  1) Why are the imports from libgmp all prefixed with '__'? That's
     a bit unusual.

  2) emacs/nt/mingw-cfg.site contains this comment:
       # We don't want to check for these functions
       # because they are implemented in libwinpthread.
     corresponding to this ChangeLog entry:

       2016-04-21  Fabrice Popineau  <fabrice.popineau@gmail.com>

               Avoid run-time dependency on libwinpthread DLL on MS-Windows

               * nt/mingw-cfg.site (ac_cv_search_clock_gettime)
               (ac_cv_func_clock_gettime, ac_cv_func_clock_settime): Force to not
               present, so that MinGW64 builds don't depend on libwinpthread.
               (Bug#22959)

     However, the binaries rely on nanosleep() from libwinpthread.
     Is it intended or unintended?
     If it is unintended, does it come from Gnulib? In this case, I'll gladly
     help to do anything needed in Gnulib to help avoid this import.

Bruno

==================== dumpbin /imports ==============================
Microsoft (R) COFF/PE Dumper Version 14.00.24210.0
Copyright (C) Microsoft Corporation.  All rights reserved.


Dump of file emacs.exe

File Type: EXECUTABLE IMAGE

  Section contains the following imports:

    ADVAPI32.dll
             400810540 Import Address Table
             40080F140 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                         407 AdjustTokenPrivileges
                         4A9 CryptAcquireContextA
                         4BA CryptGenRandom
                         563 GetUserNameA
                         596 LookupPrivilegeValueA
                         643 RegCloseKey
                         664 RegEnumValueA
                         673 RegOpenKeyExA
                         680 RegQueryValueExA

    COMCTL32.dll
             400810590 Import Address Table
             40080F190 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          68 InitCommonControls

    comdlg32.dll
             4008105A0 Import Address Table
             40080F1A0 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                           3 ChooseFontA
                           5 CommDlgExtendedError
                           A GetOpenFileNameA
                           B GetOpenFileNameW

    GDI32.dll
             4008105C8 Import Address Table
             40080F1C8 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          14 BitBlt
                          23 CombineRgn
                          2A CreateBitmap
                          31 CreateCompatibleBitmap
                          32 CreateCompatibleDC
                          33 CreateDCA
                          38 CreateDIBSection
                          41 CreateFontIndirectA
                          4E CreatePalette
                          50 CreatePen
                          54 CreateRectRgn
                          55 CreateRectRgnIndirect
                          5B CreateSolidBrush
                         181 DeleteDC
                         184 DeleteObject
                         1C4 EnumFontFamiliesExA
                         1CD EqualRgn
                         1D7 ExtTextOutA
                         1D8 ExtTextOutW
                         270 GetClipBox
                         271 GetClipRgn
                         275 GetCurrentObject
                         27D GetDeviceCaps
                         28B GetFontData
                         2AD GetObjectA
                         2B6 GetPixel
                         2D0 GetTextExtentPoint32A
                         2D1 GetTextExtentPoint32W
                         2D5 GetTextFaceA
                         2D8 GetTextMetricsA
                         2EB LineTo
                         2FF MoveToEx
                         30B PatBlt
                         321 RealizePalette
                         324 Rectangle
                         32E RestoreDC
                         335 SaveDC
                         364 SelectClipRgn
                         366 SelectObject
                         367 SelectPalette
                         36D SetBkColor
                         36E SetBkMode
                         371 SetBrushOrgEx
                         391 SetStretchBltMode
                         393 SetTextAlign
                         395 SetTextColor
                         3A2 StretchBlt
                         3A9 TranslateCharsetInfo

    libgmp-10.dll
             400810750 Import Address Table
             40080F350 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          3D __gmp_set_memory_functions
                         1F8 __gmpn_popcount
                         2DC __gmpz_add
                         2DD __gmpz_add_ui
                         2DE __gmpz_addmul
                         2DF __gmpz_addmul_ui
                         2E0 __gmpz_and
                         2E5 __gmpz_cdiv_q
                         2EE __gmpz_clear
                         2F1 __gmpz_cmp
                         2F2 __gmpz_cmp_d
                         2F4 __gmpz_cmp_ui
                         2F5 __gmpz_cmpabs
                         2F8 __gmpz_com
                         2FD __gmpz_divexact
                         304 __gmpz_export
                         306 __gmpz_fdiv_q
                         307 __gmpz_fdiv_q_2exp
                         308 __gmpz_fdiv_q_ui
                         309 __gmpz_fdiv_qr
                         30B __gmpz_fdiv_r
                         317 __gmpz_gcd
                         31A __gmpz_get_d
                         31D __gmpz_get_str
                         321 __gmpz_import
                         322 __gmpz_init
                         328 __gmpz_init_set_ui
                         32E __gmpz_ior
                         335 __gmpz_limbs_finish
                         337 __gmpz_limbs_read
                         338 __gmpz_limbs_write
                         33F __gmpz_mul
                         340 __gmpz_mul_2exp
                         342 __gmpz_mul_ui
                         34C __gmpz_pow_ui
                         358 __gmpz_roinit_n
                         35D __gmpz_scan1
                         35E __gmpz_set
                         35F __gmpz_set_d
                         362 __gmpz_set_si
                         363 __gmpz_set_str
                         364 __gmpz_set_ui
                         368 __gmpz_sizeinbase
                         36C __gmpz_sub
                         36D __gmpz_sub_ui
                         36E __gmpz_submul
                         370 __gmpz_swap
                         371 __gmpz_tdiv_q
                         374 __gmpz_tdiv_qr
                         376 __gmpz_tdiv_r
                         379 __gmpz_tdiv_ui
                         37C __gmpz_ui_pow_ui
                         380 __gmpz_xor

    KERNEL32.dll
             400810900 Import Address Table
             40080F500 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          29 BackupWrite
                          6B Beep
                          78 CancelIo
                          8D CloseHandle
                          9E CompareFileTime
                          A2 CompareStringW
                          B1 CopyFileA
                          B6 CopyFileW
                          C5 CreateEventA
                          CC CreateFileA
                          CD CreateFileMappingA
                          D4 CreateFileW
                          EA CreateProcessA
                          FC CreateThread
                         111 DebugBreak
                         11B DeleteCriticalSection
                         12B DeviceIoControl
                         12C DisableThreadLibraryCalls
                         139 DuplicateHandle
                         13F EnterCriticalSection
                         15A EnumSystemCodePagesA
                         161 EnumSystemLocalesA
                         171 ExpandEnvironmentStringsA
                         17B FillConsoleOutputAttribute
                         17C FillConsoleOutputCharacterA
                         185 FindClose
                         189 FindFirstFileA
                         190 FindFirstFileW
                         19A FindNextFileA
                         19C FindNextFileW
                         1AE FlushConsoleInputBuffer
                         1B6 FormatMessageA
                         1BB FreeLibrary
                         1C1 GenerateConsoleCtrlEvent
                         1C2 GetACP
                         1D2 GetCPInfo
                         1E5 GetCommState
                         1E6 GetCommTimeouts
                         1E7 GetCommandLineA
                         1ED GetComputerNameA
                         1FB GetConsoleCP
                         201 GetConsoleCursorInfo
                         20D GetConsoleMode
                         211 GetConsoleOutputCP
                         213 GetConsoleScreenBufferInfo
                         217 GetConsoleTitleW
                         221 GetCurrentDirectoryA
                         222 GetCurrentDirectoryW
                         228 GetCurrentProcess
                         229 GetCurrentProcessId
                         22C GetCurrentThread
                         22D GetCurrentThreadId
                         239 GetDiskFreeSpaceA
                         23C GetDiskFreeSpaceW
                         241 GetDriveTypeA
                         24C GetEnvironmentVariableA
                         250 GetExitCodeProcess
                         251 GetExitCodeThread
                         256 GetFileAttributesA
                         25B GetFileAttributesW
                         25D GetFileInformationByHandle
                         261 GetFileSize
                         264 GetFileType
                         26C GetFullPathNameA
                         26F GetFullPathNameW
                         276 GetLastError
                         279 GetLocaleInfoA
                         289 GetModuleFileNameA
                         28A GetModuleFileNameW
                         28B GetModuleHandleA
                         298 GetNamedPipeInfo
                         2AD GetNumberOfConsoleInputEvents
                         2AF GetOEMCP
                         2B0 GetOverlappedResult
                         2C6 GetProcAddress
                         2CC GetProcessHeap
                         2E0 GetProfileStringA
                         2E4 GetShortPathNameA
                         2E5 GetShortPathNameW
                         2E7 GetStartupInfoA
                         2EA GetStdHandle
                         2F3 GetSystemDefaultLCID
                         2FB GetSystemInfo
                         2FC GetSystemPowerStatus
                         301 GetSystemTimeAsFileTime
                         317 GetThreadLocale
                         31F GetTickCount
                         32C GetUserDefaultLCID
                         333 GetVersion
                         334 GetVersionExA
                         336 GetVolumeInformationA
                         338 GetVolumeInformationW
                         347 GlobalAlloc
                         34E GlobalFree
                         351 GlobalHandle
                         352 GlobalLock
                         359 GlobalUnlock
                         35F HeapAlloc
                         361 HeapCreate
                         365 HeapFree
                         369 HeapReAlloc
                         37C InitializeCriticalSection
                         397 IsDBCSLeadByteEx
                         3A5 IsValidCodePage
                         3A7 IsValidLocale
                         3D8 LeaveCriticalSection
                         3DC LoadLibraryA
                         3DF LoadLibraryW
                         3EA LocalFree
                         3FB MapViewOfFile
                         3FC MapViewOfFileEx
                         40C MultiByteToWideChar
                         42D OpenProcess
                         436 OutputDebugStringA
                         440 PeekNamedPipe
                         45B PulseEvent
                         45C PurgeComm
                         475 QueueUserAPC
                         481 RaiseException
                         486 ReadConsoleInputA
                         489 ReadConsoleInputW
                         491 ReadDirectoryChangesW
                         492 ReadFile
                         495 ReadProcessMemory
                         4BE ResetEvent
                         4C5 ResumeThread
                         4D7 ScrollConsoleScreenBufferA
                         4E3 SetCommState
                         4E4 SetCommTimeouts
                         4EA SetConsoleActiveScreenBuffer
                         4EB SetConsoleCP
                         4ED SetConsoleCtrlHandler
                         4EF SetConsoleCursorInfo
                         4F1 SetConsoleCursorPosition
                         4FD SetConsoleMode
                         502 SetConsoleOutputCP
                         505 SetConsoleScreenBufferSize
                         508 SetConsoleTitleW
                         509 SetConsoleWindowInfo
                         514 SetEndOfFile
                         519 SetErrorMode
                         51A SetEvent
                         51F SetFileAttributesA
                         522 SetFileAttributesW
                         527 SetFilePointer
                         52B SetFileTime
                         536 SetLastError
                         53F SetNamedPipeHandleState
                         540 SetPriorityClass
                         54E SetStdHandle
                         560 SetThreadLocale
                         562 SetThreadPriority
                         572 SetUnhandledExceptionFilter
                         582 Sleep
                         58A SuspendThread
                         58D SystemTimeToFileTime
                         591 TerminateProcess
                         592 TerminateThread
                         5A5 TlsGetValue
                         5B6 UnmapViewOfFile
                         5CE VirtualAlloc
                         5D1 VirtualFree
                         5D4 VirtualProtect
                         5D6 VirtualQuery
                         5DD WaitForMultipleObjects
                         5DF WaitForSingleObject
                         5E0 WaitForSingleObjectEx
                         60B WideCharToMultiByte
                         615 WriteConsoleInputA
                         618 WriteConsoleInputW
                         61B WriteConsoleOutputCharacterA
                         61F WriteFile
                         642 lstrcmpiA
                         64C lstrlenW

    MPR.dll
             400810E78 Import Address Table
             40080FA78 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                           E WNetAddConnection2A
                           F WNetAddConnection2W
                          19 WNetCloseEnum
                          24 WNetEnumResourceA
                          25 WNetEnumResourceW
                          48 WNetOpenEnumA
                          49 WNetOpenEnumW

    msvcrt.dll
             400810EB8 Import Address Table
             40080FAB8 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          38 __C_specific_handler
                          40 ___lc_codepage_func
                          43 ___mb_cur_max_func
                          52 __getmainargs
                          53 __initenv
                          54 __iob_func
                          62 __set_app_type
                          64 __setusermatherr
                          74 _acmdln
                          7C _amsg_exit
                          89 _beginthread
                          8E _cexit
                          91 _chdir
                          97 _chmod
                          9B _close
                          9D _commit
                          9E _commode
                          A7 _chdir
                          AA _creat
                          B0 _ctime64
                          BF _difftime64
                          C1 _dup
                          C2 _dup2
                          C9 _environ
                          CB _errno
                          FB _exit
                          DE _fdopen
                          E8 _fileno
                          FF _fmode
                         11C _futime64
                         127 _get_osfhandle
                         135 _getmbcp
                         136 _getpid
                         13E _gmtime64
                         14B _initterm
                         153 _hypot
                         150 _isatty
                         1BA _localtime64
                         1BB _lock
                         1C6 _lseeki64
                         1ED _mbschr
                         1FF _mbsinc
                         203 _mbslwr
                         227 _mbsncpy
                         22B _mbsnextc
                         239 _mbspbrk
                         23B _mbsrchr
                         258 _memccpy
                         25D _mkdir
                         261 _mktime64
                         267 _onexit
                         269 _open
                         26A _open_osfhandle
                         271 _pipe
                         27A _putenv
                         281 _read
                         284 _rmdir
                         29E _setjmp
                         2A2 _setmode
                         2A6 _snprintf
                         2C0 _spawnlp
                         2DA _stricmp
                         2D8 _strdup
                         2E4 _strlwr
                         2F1 _strnicmp
                         313 _sys_errlist
                         314 _sys_nerr
                         319 _time64
                         324 _tzname
                         325 _tzset
                         32F _umask
                         333 _unlink
                         334 _unlock
                         373 _wchdir
                         374 _wchmod
                         376 _wcreat
                         3C7 _wgetenv
                         3CF _wmkdir
                         3D1 _wopen
                         3DE _wrename
                         3E0 _write
                         3E1 _wrmdir
                         406 _wunlink
                         41F abort
                         41D acos
                         410 asin
                         412 atan
                         417 atof
                         418 atoi
                         421 calloc
                         424 clearerr
                         426 clock
                         42E exit
                         432 fclose
                         433 feof
                         434 ferror
                         435 fflush
                         437 fgetpos
                         439 fgets
                         442 fprintf
                         444 fputc
                         445 fputs
                         448 fread
                         449 free
                         451 fseek
                         456 fwrite
                         45A getc
                         45B getchar
                         45C getenv
                         464 isalpha
                         469 islower
                         46C isspace
                         46D isupper
                         47B isxdigit
                         47F localeconv
                         482 log10
                         485 longjmp
                         486 malloc
                         48C memchr
                         48D memcmp
                         48E memcpy
                         48F memmove
                         490 memset
                         49A putc
                         49B putchar
                         49F qsort
                         4A1 raise
                         4A3 realloc
                         4A2 rand
                         4A6 rename
                         4AC setlocale
                         4AD setvbuf
                         4AE signal
                         4B7 srand
                         4BB strcat
                         4BD strchr
                         4BE strcmp
                         4C0 strcpy
                         4C3 strerror
                         4C4 strftime
                         4C5 strlen
                         4C6 strncat
                         4C8 strncmp
                         4C9 strncpy
                         4CB strpbrk
                         4CC strrchr
                         4CD strspn
                         4CE strstr
                         4D1 strtok
                         4D3 strtol
                         4D4 strtoul
                         4DD tan
                         4E5 tolower
                         4E6 toupper
                         4E9 ungetc
                         4EC vfprintf
                         4FC wcscat
                         501 wcscpy
                         505 wcslen
                         509 wcsncpy

    ole32.dll
             4008113C0 Import Address Table
             40080FFC0 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          1E CoCreateGuid
                          55 CoInitialize
                          85 CoUninitialize
                         1FA StringFromGUID2

    libwinpthread-1.dll
             4008113E8 Import Address Table
             40080FFE8 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          10 nanosleep

    SHELL32.dll
             4008113F8 Import Address Table
             40080FFF8 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          20 DragAcceptFiles
                          21 DragFinish
                          23 DragQueryFileA
                          25 DragQueryFileW
                          26 DragQueryPoint
                          BD SHFileOperationA
                          BE SHFileOperationW
                         152 ShellExecuteExA
                         153 ShellExecuteExW
                         160 Shell_NotifyIconW

    USER32.dll
             400811450 Import Address Table
             400810050 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                           1 ActivateKeyboardLayout
                           3 AdjustWindowRect
                           B AppendMenuA
                           F AttachThreadInput
                          10 BeginDeferWindowPos
                          11 BeginPaint
                          13 BringWindowToTop
                          1F CallNextHookEx
                          31 CharLowerW
                          32 CharNextA
                          33 CharNextExA
                          35 CharPrevA
                          48 ChildWindowFromPoint
                          4C ClientToScreen
                          4E CloseClipboard
                          5E CreateCaret
                          70 CreateMenu
                          71 CreatePopupMenu
                          73 CreateWindowExA
                          A4 DefWindowProcA
                          A5 DefWindowProcW
                          A6 DeferWindowPos
                          A8 DeleteMenu
                          AB DestroyCaret
                          AE DestroyIcon
                          AF DestroyMenu
                          B1 DestroyWindow
                          B9 DispatchMessageA
                          BA DispatchMessageW
                          DB DrawTextW
                          E6 EmptyClipboard
                          EC EnableWindow
                          ED EndDeferWindowPos
                          F1 EndPaint
                          F5 EnumClipboardFormats
                         107 EnumWindows
                         10D FillRect
                         10E FindWindowA
                         111 FindWindowW
                         112 FlashWindow
                         114 FrameRect
                         11E GetAsyncKeyState
                         12D GetClassNameA
                         130 GetClientRect
                         133 GetClipboardData
                         134 GetClipboardFormatNameA
                         13E GetCursorPos
                         13F GetDC
                         142 GetDesktopWindow
                         147 GetDlgItem
                         14B GetDoubleClickTime
                         14F GetFocus
                         150 GetForegroundWindow
                         15E GetKeyNameTextA
                         160 GetKeyState
                         161 GetKeyboardLayout
                         162 GetKeyboardLayoutList
                         165 GetKeyboardState
                         16E GetMenu
                         16F GetMenuBarInfo
                         174 GetMenuItemCount
                         17C GetMessageA
                         17F GetMessageTime
                         180 GetMessageW
                         187 GetParent
                         1AF GetScrollInfo
                         1B8 GetSystemMetrics
                         1C1 GetTopWindow
                         1C4 GetUpdateRect
                         1CB GetWindow
                         1D0 GetWindowDC
                         1D4 GetWindowInfo
                         1D5 GetWindowLongA
                         1DD GetWindowPlacement
                         1DE GetWindowRect
                         1E6 GetWindowThreadProcessId
                         1EB HideCaret
                         20A InvalidateRect
                         238 KillTimer
                         241 LoadIconA
                         243 LoadImageA
                         244 LoadImageW
                         25C MapVirtualKeyA
                         260 MapWindowPoints
                         264 MessageBeep
                         265 MessageBoxA
                         273 MsgWaitForMultipleObjects
                         27D OpenClipboard
                         28A PeekMessageA
                         28B PeekMessageW
                         28E PostMessageA
                         291 PostThreadMessageA
                         2B8 RegisterClassA
                         2BB RegisterClassW
                         2C4 RegisterHotKey
                         2D6 RegisterWindowMessageA
                         2D8 ReleaseCapture
                         2D9 ReleaseDC
                         2E3 ScreenToClient
                         2E7 ScrollWindowEx
                         2EC SendInput
                         2ED SendMessageA
                         2F0 SendMessageTimeoutA
                         2F2 SendMessageW
                         2F6 SetCapture
                         2F8 SetCaretPos
                         2FE SetClipboardData
                         302 SetCursor
                         304 SetCursorPos
                         30E SetFocus
                         30F SetForegroundWindow
                         313 SetKeyboardState
                         31B SetMenu
                         325 SetParent
                         331 SetRectEmpty
                         332 SetScrollInfo
                         340 SetTimer
                         34B SetWindowLongA
                         34F SetWindowPlacement
                         350 SetWindowPos
                         354 SetWindowTextA
                         355 SetWindowTextW
                         358 SetWindowsHookExA
                         35B ShowCaret
                         35C ShowCursor
                         361 ShowWindow
                         36F SystemParametersInfoA
                         371 SystemParametersInfoW
                         376 ToAscii
                         378 ToUnicode
                         37B TrackPopupMenu
                         381 TranslateMessage
                         385 UnhookWindowsHookEx
                         386 UnionRect
                         38D UnregisterHotKey
                         3AB VkKeyScanW
                         3B8 WindowFromPoint
                         3BC keybd_event

    USP10.dll
             4008118A8 Import Address Table
             4008104A8 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                           7 ScriptFreeCache
                           8 ScriptGetCMap
                           E ScriptGetGlyphABCWidth
                          12 ScriptItemize
                          16 ScriptPlace
                          1A ScriptShape

    WINMM.dll
             4008118E0 Import Address Table
             4008104E0 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          3B mciGetErrorStringA
                          41 mciSendStringA
                          42 mciSendStringW
                          BB waveOutGetErrorTextA
                          C2 waveOutGetVolume
                          CB waveOutSetVolume

    WINSPOOL.DRV
             400811918 Import Address Table
             400810518 Import Name Table
                     0 time date stamp
                     0 Index of first forwarder reference

                          1E ClosePrinter
                          81 GetPrinterA
                          8E GetPrinterW
                          96 OpenPrinterA

  Summary

        1000 .CRT
       9B000 .bss
      476000 .data
        7000 .debug_abbrev
        1000 .debug_aranges
        4000 .debug_frame
       1F000 .debug_info
        F000 .debug_line
        6000 .debug_line_str
       13000 .debug_loclists
        1000 .debug_rnglists
        1000 .debug_str
        6000 .idata
        F000 .pdata
       3B000 .rdata
        4000 .reloc
       5C000 .rsrc
       23000 .subrs
      280000 .text
        1000 .tls
       10000 .xdata






^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-13  2:49 ` boot-time: straighten code Paul Eggert
  2023-08-13  3:26   ` Po Lu
  2023-08-14  8:02   ` boot-time: straighten code Andreas Schwab
@ 2023-08-15 21:12   ` Bruno Haible
  2023-08-16 10:13     ` Bruno Haible
  2 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-15 21:12 UTC (permalink / raw)
  To: Paul Eggert; +Cc: bug-gnulib, Emacs-devel

Paul Eggert wrote:
> I installed the attached patch into Emacs 
> master, which you should be able to test via:
> 
> 	git clone https://git.savannah.gnu.org/git/emacs.git
> 	cd emacs
> 	./autogen.sh
> 	./configure
> 	make
> 	src/emacs
> 
> Please give it a try, especially on any MS-Windows platform you happen 
> to have. I have tested only on Ubuntu 23.04 so far.
> 
> A simple way to test is to use Emacs to start editing a file (without 
> saving) and then inspect the symbolic link .#* that Emacs uses as a lock 
> file. The trailing digits of that link's contents should be the boot 
> time. These symlinks are Emacs's only use of boot time.

Here are my test results. On each of the following platforms, after
regenerating the current 'configure' file and then building the 'emacs'
directory from today, with the configure options
  --disable-silent-rules --without-all --without-x
then running "./emacs $HOME/hello.c", editing that buffer, opening
a 'M-x shell' buffer, and looking at the $HOME/.#hello.c symlink,
I can see that its last component is exactly the time_t value of the
boot time, as displayed by the gnulib 'test-readutmp' test. So, the
test passes on all these platforms:
  - Linux: Ubuntu 22.04, Alpine Linux
  - Debian GNU/Hurd 2022
  - Debian GNU/kFreeBSD 7
  - NetBSD 9.3
  - OpenBSD 7.2
  - Cygwin 2.9.0

I could not test the Windows binaries from corwin, due to problems mentioned in
<https://lists.gnu.org/archive/html/emacs-devel/2023-08/msg00543.html>.

Also, I could not test on Android (in Termux), due to a build failure, cf.
<https://github.com/termux/termux-packages/issues/6592>.

Bruno






^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-14 13:51             ` Bruno Haible
@ 2023-08-15 23:03               ` Paul Eggert
  0 siblings, 0 replies; 24+ messages in thread
From: Paul Eggert @ 2023-08-15 23:03 UTC (permalink / raw)
  To: Bruno Haible, Andreas Schwab; +Cc: bug-gnulib, Emacs-devel

On 2023-08-14 06:51, Bruno Haible wrote:
> But that is just a workaround. Is someone among the glibc people looking at the
> original glibc bughttps://sourceware.org/bugzilla/show_bug.cgi?id=30701  ?

As far as I know, nobody is looking into it other than you and me. I 
think it's low priority as people are assuming that until 2038 nobody 
should use both _TIME_BITS=64 and utmp/utmpx on 32-bit x86 or arm, and 
by 2038 (when things also stop working on 64-bit x86-64 and arm64) 
utmp/utmpx will be gone anyway.

Given the slow rate of progress in the Y2038 area I'm not sure this 
assumption is correct.



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: Windows port binaries
  2023-08-15 19:57               ` Windows port binaries Bruno Haible
@ 2023-08-16  6:45                 ` Po Lu
  2023-08-16 11:36                 ` Eli Zaretskii
  1 sibling, 0 replies; 24+ messages in thread
From: Po Lu @ 2023-08-16  6:45 UTC (permalink / raw)
  To: Bruno Haible; +Cc: Corwin Brust, Emacs-devel, Paul Eggert

Bruno Haible <bruno@clisp.org> writes:

> Corwin Brust wrote:
>> In case it helps others to assist with testing, for the moment I am
>> aggressively rebuilding the Windows port also, from (not quite every)
>> commit to emacs-29 or master, posting to:
>> 
>>   https://corwin.bru.st/emacs-29
>>   https://corwin.bru.st/emacs-30
>
> Thanks. I tried to use the
> https://corwin.bru.st/emacs-30/emacs-30-latest-no-deps.zip
> binary from today, but they don't work for me (on Windows 10), because
> they rely on two DLLs which are not contained in the 'bin' directory:
>   - libgmp-10.dll
>   - libwinpthread-1.dll
>
> Find below the output of "dumpbin.exe /imports emacs.exe".
>
> Additionally, I find it strange:
>
>   1) Why are the imports from libgmp all prefixed with '__'? That's
>      a bit unusual.

GMP prefers such a naming convention for their exported symbols.  These
symbols are defined to their programmer-facing names in gmp.h.

>   2) emacs/nt/mingw-cfg.site contains this comment:
>        # We don't want to check for these functions
>        # because they are implemented in libwinpthread.
>      corresponding to this ChangeLog entry:
>
>        2016-04-21  Fabrice Popineau  <fabrice.popineau@gmail.com>
>
>                Avoid run-time dependency on libwinpthread DLL on MS-Windows
>
>                * nt/mingw-cfg.site (ac_cv_search_clock_gettime)
>                (ac_cv_func_clock_gettime, ac_cv_func_clock_settime): Force to not
>                present, so that MinGW64 builds don't depend on libwinpthread.
>                (Bug#22959)
>
>      However, the binaries rely on nanosleep() from libwinpthread.
>      Is it intended or unintended?
>      If it is unintended, does it come from Gnulib? In this case, I'll gladly
>      help to do anything needed in Gnulib to help avoid this import.

I was under the impression that Emacs is supposed to use the nanosleep
from MinGW, so this should be investigated.



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: boot-time: straighten code
  2023-08-15 21:12   ` Bruno Haible
@ 2023-08-16 10:13     ` Bruno Haible
  0 siblings, 0 replies; 24+ messages in thread
From: Bruno Haible @ 2023-08-16 10:13 UTC (permalink / raw)
  To: Paul Eggert; +Cc: bug-gnulib, Emacs-devel

I wrote:
> So, the test passes on all these platforms:
>   - Linux: Ubuntu 22.04, Alpine Linux
>   - Debian GNU/Hurd 2022
>   - Debian GNU/kFreeBSD 7
>   - NetBSD 9.3
>   - OpenBSD 7.2
>   - Cygwin 2.9.0

It passes also on Android 11, within the Termux app.
(I now got past the build failure.)

Bruno






^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: Windows port binaries
  2023-08-15 19:57               ` Windows port binaries Bruno Haible
  2023-08-16  6:45                 ` Po Lu
@ 2023-08-16 11:36                 ` Eli Zaretskii
  2023-08-17 14:01                   ` Bruno Haible
  1 sibling, 1 reply; 24+ messages in thread
From: Eli Zaretskii @ 2023-08-16 11:36 UTC (permalink / raw)
  To: Bruno Haible; +Cc: corwin, Emacs-devel, luangruo, eggert

> From: Bruno Haible <bruno@clisp.org>
> Cc: Po Lu <luangruo@yahoo.com>, Paul Eggert <eggert@cs.ucla.edu>
> Date: Tue, 15 Aug 2023 21:57:47 +0200
> 
> >   https://corwin.bru.st/emacs-29
> >   https://corwin.bru.st/emacs-30
> 
> Thanks. I tried to use the
> https://corwin.bru.st/emacs-30/emacs-30-latest-no-deps.zip
> binary from today, but they don't work for me (on Windows 10), because
> they rely on two DLLs which are not contained in the 'bin' directory:
>   - libgmp-10.dll
>   - libwinpthread-1.dll

libwinpthread-1.dll shouldn't be there.  But maybe there's no way to
avoid that with MinGW64 ports of GCC.  See below.

> Find below the output of "dumpbin.exe /imports emacs.exe".

A more GNU-friendly way is

  objdump -p emacs.exe | fgrep "DLL Name:"

> Additionally, I find it strange:
> 
>   1) Why are the imports from libgmp all prefixed with '__'? That's
>      a bit unusual.

Why are you asking us?  The Emacs project doesn't build libgmp, it
only uses it.  This question shouldf go to the GMP developers, IMO.

>   2) emacs/nt/mingw-cfg.site contains this comment:
>        # We don't want to check for these functions
>        # because they are implemented in libwinpthread.
>      corresponding to this ChangeLog entry:
> 
>        2016-04-21  Fabrice Popineau  <fabrice.popineau@gmail.com>
> 
>                Avoid run-time dependency on libwinpthread DLL on MS-Windows
> 
>                * nt/mingw-cfg.site (ac_cv_search_clock_gettime)
>                (ac_cv_func_clock_gettime, ac_cv_func_clock_settime): Force to not
>                present, so that MinGW64 builds don't depend on libwinpthread.
>                (Bug#22959)
> 
>      However, the binaries rely on nanosleep() from libwinpthread.
>      Is it intended or unintended?

I guess this is some kind of regression.  But I'm not sure we can
avoid this, when MinGW64 is used.  (I use mingw.org's MinGW, and my
Emacs on Windows doesn't have the libwinpthread dependency.)



^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: Windows port binaries
  2023-08-16 11:36                 ` Eli Zaretskii
@ 2023-08-17 14:01                   ` Bruno Haible
  2023-08-17 14:14                     ` Eli Zaretskii
  0 siblings, 1 reply; 24+ messages in thread
From: Bruno Haible @ 2023-08-17 14:01 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: corwin, Emacs-devel, luangruo, eggert

Eli Zaretskii wrote:
> > they rely on two DLLs which are not contained in the 'bin' directory:
> >   - libgmp-10.dll
> >   - libwinpthread-1.dll
> 
> libwinpthread-1.dll shouldn't be there. ...
> But I'm not sure we can avoid this, when MinGW64 is used.

The dependency comes from Gnulib. Emacs uses the gnulib module 'nanosleep',
and it picks the nanosleep function from libwinpthread-1.dll. I can reproduce
this directly in Gnulib.

Since this DLL dependency is an annoyance also for other people who produce
Windows binaries (I'm thinking of gettext and others), I'll change Gnulib
to avoid this dependency.

> A more GNU-friendly way is
> 
>   objdump -p emacs.exe | fgrep "DLL Name:"

Thanks; that's a nice shortcut, because I usually have 'objdump' in my $PATH.

Bruno






^ permalink raw reply	[flat|nested] 24+ messages in thread

* Re: Windows port binaries
  2023-08-17 14:01                   ` Bruno Haible
@ 2023-08-17 14:14                     ` Eli Zaretskii
  0 siblings, 0 replies; 24+ messages in thread
From: Eli Zaretskii @ 2023-08-17 14:14 UTC (permalink / raw)
  To: Bruno Haible; +Cc: corwin, Emacs-devel, luangruo, eggert

> From: Bruno Haible <bruno@clisp.org>
> Cc: corwin@bru.st, Emacs-devel@gnu.org, luangruo@yahoo.com, eggert@cs.ucla.edu
> Date: Thu, 17 Aug 2023 16:01:23 +0200
> 
> Eli Zaretskii wrote:
> > > they rely on two DLLs which are not contained in the 'bin' directory:
> > >   - libgmp-10.dll
> > >   - libwinpthread-1.dll
> > 
> > libwinpthread-1.dll shouldn't be there. ...
> > But I'm not sure we can avoid this, when MinGW64 is used.
> 
> The dependency comes from Gnulib. Emacs uses the gnulib module 'nanosleep',
> and it picks the nanosleep function from libwinpthread-1.dll. I can reproduce
> this directly in Gnulib.
> 
> Since this DLL dependency is an annoyance also for other people who produce
> Windows binaries (I'm thinking of gettext and others), I'll change Gnulib
> to avoid this dependency.

Thanks, but please allow MinGW to use its own implementation if time.h
declares it and a test program calling it links without pthreads.  At
least mingw.org's MinGW does have nanosleep which doesn't need
pthreads.



^ permalink raw reply	[flat|nested] 24+ messages in thread

end of thread, other threads:[~2023-08-17 14:14 UTC | newest]

Thread overview: 24+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
     [not found] <4536176.VaOIPsP7d9@nimes>
2023-08-13  2:49 ` boot-time: straighten code Paul Eggert
2023-08-13  3:26   ` Po Lu
2023-08-13  6:35     ` Paul Eggert
2023-08-13 13:45     ` Bruno Haible
2023-08-13 14:16     ` Bruno Haible
2023-08-13 14:36     ` Bruno Haible
2023-08-13 23:44       ` Po Lu
2023-08-13 23:59         ` Bruno Haible
2023-08-14  1:07           ` Po Lu
2023-08-14  2:14             ` Corwin Brust
2023-08-15 19:57               ` Windows port binaries Bruno Haible
2023-08-16  6:45                 ` Po Lu
2023-08-16 11:36                 ` Eli Zaretskii
2023-08-17 14:01                   ` Bruno Haible
2023-08-17 14:14                     ` Eli Zaretskii
2023-08-14  8:02   ` boot-time: straighten code Andreas Schwab
2023-08-14  9:15     ` Bruno Haible
2023-08-14  9:20       ` Andreas Schwab
2023-08-14 10:19         ` Bruno Haible
2023-08-14 10:33           ` Andreas Schwab
2023-08-14 13:51             ` Bruno Haible
2023-08-15 23:03               ` Paul Eggert
2023-08-15 21:12   ` Bruno Haible
2023-08-16 10:13     ` Bruno Haible

Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).