Jak touch -t
polecenie działa dokładnie wewnętrznie (próbowałem znaleźć jego kod źródłowy, ale nie mogłem)?
źródło
Jak touch -t
polecenie działa dokładnie wewnętrznie (próbowałem znaleźć jego kod źródłowy, ale nie mogłem)?
touch
wywołuje utimes
wywołanie systemowe, aby ustawić czas modyfikacji pliku i czas jego dostępu. W niektórych systemach zamiast utimes
otwiera plik, a następnie ustawia czas pliku za pomocą deskryptora, np utimensat
. W systemie Linux.
Możesz zobaczyć, jak touch
działa twój system, patrząc na wywołania systemowe, które wykonuje. W Linuksie użyj strace , np strace touch -d '1 hour ago' foo
.
Gdzie znaleźć kod źródłowy, zależy od systemu operacyjnego. Wersja GNU jest w coreutils , jest wersja w głównym drzewie źródłowym dowolnego BSD, jest wersja w BusyBox , w Minix itp.
Czasami nawet nie potrzebujesz kodu źródłowego. Zastosowanie 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 +++
Dlaczego cześć tam utimensat()
. Czym jesteś?
$ man utimensat
NAME
utimensat, futimens - change file timestamps with nanosecond precision
Jest więc funkcja, która zmienia znaczniki czasu pliku i touch
używa go do aktualizacji znacznika czasu pliku. I tak to działa wewnętrznie.
Oto, jak to działa w systemie Solaris. truss
jest używana zamiast tego strace
jest tu całkiem inne polecenie.
Podobnie jak w Gnu / Linux, używane utimensat
jest wywołanie systemowe.
$ 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)