Как touch -t
команда работает точно, внутренне (я пытался найти ее исходный код, но не смог)?
источник
Как touch -t
команда работает точно, внутренне (я пытался найти ее исходный код, но не смог)?
touch
вызывает utimes
системный вызов, чтобы установить время изменения файла и время его доступа. В некоторых системах вместо этого utimes
он открывает файл, а затем устанавливает время файла через дескриптор, например, utimensat
в Linux.
Вы можете увидеть, как touch
работает ваша система, посмотрев на системные вызовы, которые она выполняет. Под Linux используйте strace , например strace touch -d '1 hour ago' foo
.
Где найти исходный код зависит от вашей операционной системы. Версия GNU находится в coreutils , есть версия в главном дереве исходного кода любого BSD, есть версия в BusyBox , в Minix и т. Д.
Иногда вам даже не нужен исходный код. Использование strace
.
$ strace touch -t 201212121212 foobar
execve("/usr/bin/touch", ["touch", "-t", "201212121212", "foobar"], [/* 61 vars */]) = 0
[...] lots of noise [...]
open("foobar", O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK, 0666) = 3
dup2(3, 0) = 0
close(3) = 0
utimensat(0, NULL, {{1355310720, 0}, {1355310720, 0}}, 0) = 0
close(0) = 0
close(1) = 0
close(2) = 0
exit_group(0) = ?
+++ exited with 0 +++
Почему привет utimensat()
? Что ты такое?
$ man utimensat
NAME
utimensat, futimens - change file timestamps with nanosecond precision
Так что есть функция, которая изменяет временные метки файла и touch
использует ее для обновления временной метки файла. И вот как это работает внутри.
Вот как это работает на Solaris. truss
вместо этого strace
используется совершенно другая команда.
Как и в Gnu / Linux, utimensat
используется системный вызов.
$ truss -vall -u a.out -f touch -t 1306080000 z
4160: execve("/usr/bin/touch", 0xF0770FC0, 0xF0770FD4) argc = 4
...
4160/1@1: -> main(0x4, 0xf0770fc0, 0xf0770fd4, 0xf0770f7c)
...
4160/1@1: -> atoi_for2(0xf0771131, 0x0, 0x24, 0xebc95be0)
4160/1@1: <- atoi_for2() = 13
4160/1@1: -> atoi_for2(0xf0771133, 0x0, 0x24, 0xebc95be0)
4160/1@1: <- atoi_for2() = 6
4160/1@1: -> atoi_for2(0xf0771135, 0x0, 0x24, 0xebc95be0)
4160/1@1: <- atoi_for2() = 8
4160/1@1: -> atoi_for2(0xf0771137, 0x0, 0x24, 0xebc95be0)
4160/1@1: <- atoi_for2() = 0
4160/1@1: -> atoi_for2(0xf0771139, 0x0, 0x24, 0xebc95be0)
4160/1@1: <- atoi_for2() = 0
4160/1@1: <- parse_time() = 0x51b257e0
4160/1: stat64("z", 0xF0770ED0) = 0
4160/1: d=0x08A00002 i=75783706 m=0100644 l=1 u=100 g=10 sz=0
4160/1: at = Jun 8 01:48:08 CEST 2013 [ 1370648888.022270973 ]
4160/1: mt = Jun 8 01:48:08 CEST 2013 [ 1370648888.022270973 ]
4160/1: ct = Jun 8 01:48:08 CEST 2013 [ 1370648888.022273810 ]
4160/1: bsz=4096 blks=0 fs=tmpfs
4160/1: utimensat(AT_FDCWD, "z", 0xF0770F60, 0) = 0
4160/1: at = Jun 8 00:00:00 CEST 2013 [ 1370642400.000000000 ]
4160/1: mt = Jun 8 00:00:00 CEST 2013 [ 1370642400.000000000 ]
4160/1@1: <- main() = 0
4160/1@1: -> _fini()
4160/1@1: <- _fini() = 0xebcc0140
4160/1: _exit(0)