summaryrefslogtreecommitdiff
path: root/minix/lib/libc/sys/utimensat.c
blob: c298ec60cab2086b8523d65fc7a96b6c1d9e188b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
#include <sys/cdefs.h>
#include "namespace.h"
#include <lib.h>

#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <limits.h>
#include <errno.h>

/* Implement a very large but not complete subset of the utimensat()
 * Posix:2008/XOpen-7 function.
 * Are handled the following cases:
 * . utimensat(AT_FDCWD, "/some/absolute/path", , )
 * . utimensat(AT_FDCWD, "some/path", , )
 * . utimensat(fd, "/some/absolute/path", , ) although fd is useless here
 * Are not handled the following cases:
 * . utimensat(fd, "some/path", , ) path to a file relative to some open fd
 */
int utimensat(int fd, const char *name, const struct timespec tv[2],
    int flags)
{
  message m;
  static const struct timespec now[2] = { {0, UTIME_NOW}, {0, UTIME_NOW} };

  if (tv == NULL) tv = now;

  if (name == NULL) {
	errno = EINVAL;
	return -1;
  }
  if (name[0] == '\0') { /* POSIX requirement */
	errno = ENOENT;
	return -1;
  }
  if (fd != AT_FDCWD && name[0] != '/') { /* Not supported */
	errno = EINVAL;
	return -1;
  }

  if ((unsigned)flags > SHRT_MAX) {
	errno = EINVAL;
	return -1;
  }

  memset(&m, 0, sizeof(m));
  m.m_vfs_utimens.len = strlen(name) + 1;
  m.m_vfs_utimens.name = __UNCONST(name);
  m.m_vfs_utimens.atime = tv[0].tv_sec;
  m.m_vfs_utimens.mtime = tv[1].tv_sec;
  m.m_vfs_utimens.ansec = tv[0].tv_nsec;
  m.m_vfs_utimens.mnsec = tv[1].tv_nsec;
  m.m_vfs_utimens.flags = flags;

  return(_syscall(VFS_PROC_NR, VFS_UTIMENS, &m));
}