Skąd „uname” wyciąga informacje?


Odpowiedzi:


31

unameużywa wywołania systemowego, uname(2)aby wyświetlić wyświetlane informacje dotyczące jądra.

Streszczenie to:

#include <sys/utsname.h>
int uname(struct utsname *buf);

gdzie uname(2)zwraca informacje w strukturze wskazanej przez buf. Ponadto można odczytać pliku nagłówka utsname.hz /usr/include/"$(arch)"-linux-gnu/sys/utsname.hkopać głębiej.

Sprawdź, man 2 unameaby uzyskać więcej informacji na ten temat.


kiedy uruchamiam „uname -i”, wynikiem jest „x86_64”. Kiedy odsyłam do „/usr/include/x86_64-linux-gnu/sys/utsname.h”, nie widzę niczego, co odnosi się do „x86_64”. Nawiązałem do „uname man 2” i stwierdza, że ​​część informacji o nazwie utsname jest przywoływana przez „/ proc / sys / kernel / {ostype}, {hostname}, {osrelease}, {version} i {domainname}” problem polega na tym, że że żaden z tych plików nie zawiera odniesienia do „x86_64”. Jakieś inne rekomendacje?
Roy Hernandez

@RoyHernandez Jaka jest wydajność locate --regex '^/usr/include/.*/sys/utsname.h$'?
heemayl

Dane wyjściowe to: „/usr/include/x86_64-linux-gnu/sys/utsname.h”
Roy Hernandez

@RoyHernandez To mówi, że plik istnieje i robiłeś coś złego ...
heemayl

Kiedy uruchamiam uname -iwyjście jest x86_64. Po uruchomieniu locate --regex '^/usr/include/.*/sys/utsname.h$'zwracane dane wyjściowe/usr/include/x86_64-linux-gnu/sys/utsname.h
Roy Hernandez

22

Program stracepozwala nam przeglądać wywołania systemowe, które może wykonywać aplikacja. Ze uname -ajest to oczywiste, że tylko openpołączenia pójść do bibliotek systemowych, więc technicznie nie ma pliku w systemie plików, że unameotwiera się na czytaniu. Raczej wykonuje wywołania systemowe za pomocą bibliotek C.

Jak heemayl słusznie zauważył, istnieje wywołanie sys do pobierania informacji przechowywanych w unamestrukturze. Jest to strona podręcznika, sugeruje następujące:

Jest to wywołanie systemowe, a system operacyjny prawdopodobnie zna jego nazwę, wydanie i wersję. . . . . . Część informacji o nazwie utsname jest również dostępna przez / proc / sys / kernel / {ostype, nazwa hosta, osrelease, wersja, nazwa domeny}.

Część informacji o nazwie utsname jest również dostępna przez / proc / sys / kernel / {ostype, nazwa hosta, osrelease, wersja, nazwa domeny}.

/procsystem plików jest jednak wirtualny, co oznacza, że ​​istnieje tylko podczas działania systemu operacyjnego. Dlatego w pewnym stopniu jest on ustawiony w bibliotekach jądra lub bibliotek systemowych.

Na koniec, czytając kod źródłowy, za pomocą uname.cktórego można uzyskać apt-get source coreutils, możemy zauważyć, że rzeczywiście korzysta on z utsname.hbiblioteki (wydrukowanej z numerami wierszy):

 19 
 20 #include <config.h>
 21 #include <stdio.h>
 22 #include <sys/types.h>
 23 #include <sys/utsname.h>
 24 #include <getopt.h>
 25 

strace wydajność:

skolodya@ubuntu:$ strace uname -a
execve("/bin/uname", ["uname", "-a"], [/* 58 vars */]) = 0
brk(0)                                  = 0x1478000
access("/etc/ld.so.nohwcap", F_OK)      = -1 ENOENT (No such file or directory)
mmap(NULL, 8192, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7efee6935000
access("/etc/ld.so.preload", R_OK)      = -1 ENOENT (No such file or directory)
open("/etc/ld.so.cache", O_RDONLY|O_CLOEXEC) = 3
fstat(3, {st_mode=S_IFREG|0644, st_size=137226, ...}) = 0
mmap(NULL, 137226, PROT_READ, MAP_PRIVATE, 3, 0) = 0x7efee6913000
close(3)                                = 0
access("/etc/ld.so.nohwcap", F_OK)      = -1 ENOENT (No such file or directory)
open("/lib/x86_64-linux-gnu/libc.so.6", O_RDONLY|O_CLOEXEC) = 3
read(3, "\177ELF\2\1\1\0\0\0\0\0\0\0\0\0\3\0>\0\1\0\0\0\320\37\2\0\0\0\0\0"..., 832) = 832
fstat(3, {st_mode=S_IFREG|0755, st_size=1840928, ...}) = 0
mmap(NULL, 3949248, PROT_READ|PROT_EXEC, MAP_PRIVATE|MAP_DENYWRITE, 3, 0) = 0x7efee6350000
mprotect(0x7efee650b000, 2093056, PROT_NONE) = 0
mmap(0x7efee670a000, 24576, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED|MAP_DENYWRITE, 3, 0x1ba000) = 0x7efee670a000
mmap(0x7efee6710000, 17088, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED|MAP_ANONYMOUS, -1, 0) = 0x7efee6710000
close(3)                                = 0
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7efee6912000
mmap(NULL, 8192, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7efee6910000
arch_prctl(ARCH_SET_FS, 0x7efee6910740) = 0
mprotect(0x7efee670a000, 16384, PROT_READ) = 0
mprotect(0x606000, 4096, PROT_READ)     = 0
mprotect(0x7efee6937000, 4096, PROT_READ) = 0
munmap(0x7efee6913000, 137226)          = 0
brk(0)                                  = 0x1478000
brk(0x1499000)                          = 0x1499000
open("/usr/lib/locale/locale-archive", O_RDONLY|O_CLOEXEC) = 3
fstat(3, {st_mode=S_IFREG|0644, st_size=7216688, ...}) = 0
mmap(NULL, 7216688, PROT_READ, MAP_PRIVATE, 3, 0) = 0x7efee5c6e000
close(3)                                = 0
uname({sys="Linux", node="eagle", ...}) = 0
fstat(1, {st_mode=S_IFCHR|0620, st_rdev=makedev(136, 0), ...}) = 0
mmap(NULL, 4096, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0) = 0x7efee6934000
uname({sys="Linux", node="eagle", ...}) = 0
uname({sys="Linux", node="eagle", ...}) = 0
write(1, "Linux eagle 4.1.0-040100rc2-gene"..., 113Linux eagle 4.1.0-040100rc2-generic #201505032335 SMP Mon May 4 03:36:35 UTC 2015 x86_64 x86_64 x86_64 GNU/Linux
) = 113
close(1)                                = 0
munmap(0x7efee6934000, 4096)            = 0
close(2)                                = 0
exit_group(0)                           = ?
+++ exited with 0 +++

kiedy uruchamiam „uname -i”, wynikiem jest „x86_64”. Kiedy odsyłam do „/usr/include/x86_64-linux-gnu/sys/utsname.h”, nie widzę niczego, co odnosi się do „x86_64”. Nawiązałem do „uname man 2” i stwierdza, że ​​część informacji o nazwie utsname jest przywoływana przez „/ proc / sys / kernel / {ostype}, {hostname}, {osrelease}, {version} i {domainname}” problem polega na tym, że że żaden z tych plików nie zawiera odniesienia do „x86_64”. Jakieś inne rekomendacje?
Roy Hernandez

@RoyHernandez W C można na przykład określić architekturę procesora na podstawie wielkości, jaką przyjmuje liczba całkowita - patrz tutaj . Więc uname.cniekoniecznie trzeba do tego używać biblioteki - oczywiście możemy spojrzeć na kod źródłowy.
Sergiy Kolodyazhnyy

W rzeczywistości opiera się na bibliotece. . . machine.h
Sergiy Kolodyazhnyy

machine.hwydaje się być obrzucony całym systemem. Na jakim machine.hpliku polega?
Roy Hernandez

@ RoyHernandez wszystkie wymienione machine.hw moim systemie wydają się znajdować w /usr/src/linux-headers-3.19.0-33katalogu. Jest bardzo prawdopodobne, że korzysta z biblioteki dostarczanej przez obecnie działające jądro
Sergiy Kolodyazhnyy

6

Oczywiście odpowiedź heemayla jest poprawna.

Dla zabawy, oto działający fragment kodu C przedstawiający dane zwrócone przez uname()(rodzaj domowej roboty, unamejeśli chcesz): skompiluj go gcc uname.c -o unamei uruchom za pomocą ./uname:

#include <stdio.h> // printf()
#include <sys/utsname.h> // uname()

int main() {
        int ret; // stores the return value of uname()
        struct utsname utsname; // stores the data returned by uname()
        struct utsname *utsname_ptr = &utsname; // pointer to the struct holding the data returned by uname()

        ret = uname(utsname_ptr); // calls uname() on utsname_ptr and stores its return value in ret

        /* prints the fields of utsname */

        printf("%s\n", utsname.sysname);
        printf("%s\n", utsname.nodename);
        printf("%s\n", utsname.release);
        printf("%s\n", utsname.version);
        printf("%s\n", utsname.machine);

        /* returns the return value of uname() */

        return(ret);
}
% ./uname 
Linux
user-X550CL
4.2.0-25-generic
#30-Ubuntu SMP Mon Jan 18 12:31:50 UTC 2016
x86_64

skąd bierze się printf("%\n", utsname.machine);informacja?
Roy Hernandez

@RoyHernandez Z struktury utsname, która jest wypełniana podczas połączenia z uname(). Przykład prawdopodobnie nie jest zbyt prosty dla kogoś bez podstaw języka C, ale mniej więcej to, co się dzieje: zadeklarowano (typ structdanych C) typu o utsnamenazwie utsname(typ zdefiniowany w <sys/utsname.h>); następnie wskaźnik do niego nazwany utsname_ptrjest zadeklarowana (ponieważ uname()przyjmuje wskaźnik do structtypu utsnamejako argument, choć można było uniknąć w tym przypadku, ale to inna historia).
Kos

Następnie wywołanie do uname()powoduje wypełnienie struktury utsname, która w momencie printf()wywołania zawiera różne wartości w różnych polach. Niestety, jeśli nie znasz C, prawdopodobnie nie będzie to łatwe do szczegółowego zrozumienia, ale chodzi o to, że uname()wypełnia celowo zbudowaną strukturę danych, której pola są później drukowane printf().
Kos

4

Jako dodatek do odpowiedzi heemayla możesz uzyskać pewne informacje jak w unamepoleceniu z /proc/version.


/ proc / wersja zawiera „Linux wersja 3.19.0-47-generic (buildd @ lgw01-19) (gcc wersja 4.8.2 (Ubuntu 4.8.2-19ubuntu1)) # 53 ~ 14.04.1-Ubuntu SMP Pn 18 stycznia 16 : 09: 14 UTC 2016 ”i„ uname -i ”wyjście to„ x86_64 ”.
Roy Hernandez
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.