[libc] [llvm] [libc] Implemented utimes (Issue #133953) (PR #134167)

Michael Jones via llvm-commits llvm-commits at lists.llvm.org
Wed Apr 2 16:38:42 PDT 2025


================
@@ -0,0 +1,80 @@
+//===-- Linux implementation of utimes -------------------------------------===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===/
+
+#include "src/sys/time/utimes.h"
+
+#include "hdr/types/struct_timeval.h"
+
+#include "src/__support/OSUtil/syscall.h"
+#include "src/__support/common.h"
+
+#include "src/errno/libc_errno.h"
+
+#include <cerrno>
+#include <sys/syscall.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+
+namespace LIBC_NAMESPACE_DECL {
+  
+  LLVM_LIBC_FUNCTION(int, utimes, (const char *path, const struct timeval times[2])) {
+    int ret;
+  
+  #ifdef SYS_utimensat
+    //the utimensat syscall requires a timespec struct, not timeval.
+    struct timespec ts[2];
+    struct timespec *ts_ptr = nullptr; // default value if times is NULL
+    
+    // convert the microsec values in timeval struct times 
+    // to nanosecond values in timespec struct ts
+    if (times != NULL) {
+      
+      // ensure consistent values
+      if ((times[0].tv_usec < 0 || times[1].tv_usec < 0) || 
+         (times[0].tv_usec >= 1000000 || times[1].tv_usec >= 1000000)) {
+           libc_errno = EINVAL;
+           return -1;
+      }
+      
+      // set seconds in ts
+      ts[0].tv_sec = times[0].tv_sec; 
+      ts[1].tv_sec = times[1].tv_sec;
+      
+      // convert u-seconds to nanoseconds
+      ts[0].tv_nsec = times[0].tv_usec * 1000;
+      ts[1].tv_nsec = times[1].tv_usec * 1000;
+      
+      ts_ptr = ts;
+    }
+    
+    // If times was NULL, ts_ptr remains NULL, which utimensat interprets
+    // as setting times to the current time.
+  
+    // utimensat syscall. 
+    // flags=0 means don't follow symlinks (like utimes)
+    ret = LIBC_NAMESPACE::syscall_impl<int>(SYS_utimensat, AT_FDCWD, path,
+                                             ts_ptr, 0);
+  
+  #elif defined(SYS_utimes)
+    // No need to define a timespec struct, use the syscall directly.
+    ret = LIBC_NAMESPACE::syscall_impl<int>(SYS_utimes, path, times);
+  #else
+  #error "utimensat and utimes syscalls not available."
+    // To avoid compilation errors when neither is defined, return an error.
+    libc_errno = ENOSYS; // Function not implemented
+    return -1;
----------------
michaelrj-google wrote:

the `#error` already causes a compilation error. This part isn't necessary.

https://github.com/llvm/llvm-project/pull/134167


More information about the llvm-commits mailing list