1 W SYSTEMIE OPERACYJNYM UNIX / LINUXPODSTAWY PROGRAMOWANIA W SYSTEMIE OPERACYJNYM UNIX / LINUX
2 Plan wykładu Edytowanie vi / vim Kompilowanie za pomocą GCCWykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek
3 Edytowanie vi / vim Edytowanie vi / vim Kompilowanie za pomocą GCCWykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek
4 vi - uruchomienie vi tworzony jest nowy plik tymczasowy, bez nazwy nazwa; vi plik1 plik2 … do edycji otwierane są pliki z listy argumentów wywołania edytora vi (przełączanie :e); vi –R plik1 pliki plik1 otwierany „tylko do odczytu”; edycja tylko na buforze, zapis pliku trzeba wymuszać( np. :w!); vi –r wyświetlana jest lista odnalezionych plików wymiany; vi –r plik pozwala na odzyskanie pliku wymiany plik (taki sam efekt :recovery) T.Przechlewski, Ściąga do vi
5 vi – podstawowe operacje na plikachZZ lub :x wyjście z zapisaniem do bieżącego pliku :w zapisanie zmian do bieżącego pliku :q wyjście, jeśli plik nie był modyfikowany :q! wyjście bez zapisywania zmian :w plik zapisanie zawartości bufora do pliku :x,yw plik zapisanie od wiersza x do y do pliku :e plik rozpoczęcie edycji pliku :e! pominięcie dokonanych modyfikacji i udostępnienie ponownie bieżącego pliku do edycji :r plik wstawienie zawartości pliku za bieżącym wierszem :n edycja następnego pliku z listy argumentów :N edycja poprzedniego pliku z listy argumentów :f plik nadanie bieżącemu plikowi nazwę plik :ls wyświetlenie listy aktualnie edytowanych plików
6 vi –przejście do trybu wprowadzania tekstui wstawianie tekstu przed kursorem I przesuwa kursor do początku wiersza bieżącego i przechodzi do trybu wpisywania o tworzy wiersz poniżej bieżącego, zaczyna tryb wpisywania O tworzy wiersz powyżej bieżącego, zaczyna tryb wpisywania a wstawianie tekstu za bieżąca pozycja kursora A przesuwa kursor na koniec wiersza bieżącego, zaczyna trybu wpisywania rc wstawienie znaku c w pozycji kursora R przejście do trybu nadpisywania tekstu
7 vi – przesuwanie kursora^ początek wiersza z lewej $ koniec wiersza z prawej + lub
8 vi –usuwanie i modyfikowanie tekstuDo usuwania fragmentów tekstu służy polecenie dzakres, gdzie zakres określa, co zostanie usunięte. Zakres określamy dodając do polecenia d polecenie przesunięcia kursora, m.in.: nx usuwa n znaków na prawo od kursora nX usuwa n znaków na lewo od kursora D lub d$ usuwa od kursora do końca wiersza dd usuwa bieżący wiersz ndd lub dnd usuwa n wierszy począwszy od bieżącego d0 usuwa od początku wiersza do kursora dw usuwa od kursora do początku następnego wyrazu dG usuwa od bieżącego wiersza do końca pliku d/napis
9 vi – przesuwanie tekstu z / do buforaUsuwany fragment tekstu jest kopiowany do specjalnego bufora. Zawartość tego bufora może zostać wstawiona w innym miejscu za pomocą polecenia p lub P. Polecenie y, o składni podobnej do poleceń d czy c, kopiuje fragment tekstu określony przez zakres do bufora nie nazwanego. d, dd lub D usuwa tekst do bufora yy lub Y kopiuje bieżący wiersz do bufora nyy lub nY kopiuje n wierszy do bufora, począwszy od bieżącego ynw kopiuje n kolejnych słów do bufora y/tekst kopiuje od kursora do tekstu do bufora p wstawia zawartość bufora za wiersz bieżący P jak wyżej, ale przed wiersz bieżący d, y tekst bufor p
10 vi – operacje wyszukiwania i zamiany/napis
11 vi – konfigurowanie edytora:set ai automatyczne wcinanie wiersza na głębokość wcięcia wiersza poprzedniego (noai – wyłączenie) :set nu numerowanie wierszy (nonu – wyłączenie) :set ts=n rozmiar wcięcia po naciśnięciu
12 Kompilowanie za pomocą GCCEdytowanie vi / vim Kompilowanie za pomocą GCC Wykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek
13 GCC GCC (ang. GNU Compiler Collection) jest zestawem kompilatorów kompilatory m.in. gcc, g++, g77 (fortran), itd. Składnia (dla C i C++) gcc [opcje | pliki] – kompilator C g++ [opcje | pliki] – kompilator C++ Etapy działania GCC preprocesorowanie (ang. preprocessing) kompilacja (ang. compilation) asemblacja (ang. assembling) konsolidacja (ang. linking) K.Wall, Using the GNU Compiler Collection, SAMS
14 GCC – rozpoznawane rozszerzenia plików.c źródła C .C źródła C++ .cc źródła C++ .cxx źródła C++ .c++ źródła C++ .i preprocessed C .ii preprocessed C++ .s źródła asemblera .S źródła asemblera .h pliki nagłówkowe .o pliki typu obiekt .a biblioteki .so biblioteki
15 GCC – kompilacja jednego źródła$ ls –l -rw-r--r-- 1 burek users :04 prostokat.c $ gcc prostokat.c $ ls -l -rwxr-xr-x 1 burek users :05 a.out $ ./a.out obwod = 60, pole = 200 $ gcc prostokat.c -o prostokat -rwxr-xr-x 1 burek users :11 prostokat $ ./prostokat
16 GCC – główne opcje -E Tylko preprocesing, na wyjściu dostajemy pliki źródłowe przerobione przez preprocesor. -S Zatrzymuj po poziomie kompilacji, nie assembluj, na wyjściu mamy plik źródłowym z kodem assemblera. -c Zatrzymuj po poziomie kompilacji lub asemblacji, bez linkowania. Na wyjściu dostajemy pliki typu obiekt dla każdego pliku źródłowego. -o nazwa Wskazanie nazwy pliku wynikowego dla danej operacji (przy kompilacji standardowo dostajemy a.out).
17 GCC – kompilacja wielu źródełW prostokat.c jest f-cja main(), w lib.c funkcje wywoływane z main. $ gcc prostokat.c lib.c -o prostokat ; ls -l -rw-r--r-- 1 burek users :07 lib.c -rwxr-xr-x 1 burek users :08 prostokat -rw-r--r-- 1 burek users :04 prostokat.c $ gcc –c prostokat.c lib.c ; ls -l -rw-r--r-- 1 burek users :10 lib.o -rw-r--r-- 1 burek users :10 prostokat.o $ gcc prostokat.o lib.o -o prostokat ; ls -l -rwxr-xr-x 1 burek users :11 prostokat
18 GCC – inne opcje (1) Opcje preprocesora: Opcje debugera:-D macro Ustaw macro na 1. -D macro=defn Zdefinjuj makro macro jako defn. -U macro Skasuj definicje makra macro. Opcje debugera: -g Dodatkowe informacje dla debugera. Kompilacja z tą opcją pozwala na późniejsze debugowanie programu. -ggdb Dodatkowe informacje dla DBG (możliwość wykorzystania rozszerzeń GDB) Opcje preprocesora: -I dir Dodaje katalog dir do listy katalogów przeszukiwanych ze względu na pliki nagłówkowe (include file). -L dir Dodaje katalog dir do listy katalogów przeszukiwanych przy użyciu przełącznika -l (patrz opcje linkera)
19 GCC – inne opcje (2) Opcje linkera: Opcje optymalizacji:-llibrary Użyj biblioteki library kiedy linkujesz. Uwaga! Gcc automatycznie dodaje przedrostel lib i końcówkę .a, np. -lFOX w celu załadowania libFOX.a. Patrz też -L. -nostdlib Nie używaj standardowych bibliotek systemowych i startowych plików kiedy linkujesz. Używaj tylko wskazane. Opcje optymalizacji: -O Optymalizacja. -Olevel Poziom optymalizacji: 0,1,2,3, jeśli 0, to brak optymalizacji. Opcje ostrzeżeń: -Wall Wypisuje ostrzeżenia dla wszystkich sytuacji, które pretendują do konstrukcji, których używania się nie poleca i których użycie jest proste do uniknięcia, nawet w połączeniu z makrami.
20 GCC – przykłady użycia $ gcc –c –O2 prostokat.c$ gcc –c -I ../include prostakat.c $ gcc prostokat.c –L ../libs –ltest $ gcc –c –D DEBUG prostokat.c $ gcc –g prostokat.c –o prostokat
21 GCC – interpretacja komunikatów o błędach$ gcc prostokat.c prostokat.c: In function `main': prostokat.c:10: error: `obwod' undeclared (first use in this function) prostokat.c:10: error: (Each undeclared identifier is reported only once prostokat.c:10: error: for each function it appears in.) prostokat.c:11: error: `pole' undeclared (first use in this function) prostokat.c:16:2: warning: no newline at end of file prostokat.c #include
22 Wykorzystanie GNU MakeEdytowanie vi / vim Kompilowanie za pomocą GCC Wykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek
23 Budowa prostego pliku Makefileoutprogname : binarytree.o mainprog.o gcc -o outprogname binarytree.o mainprog.o binarytree.o : binarytree.c gcc -c binarytree.c mainprog.o : mainprog.c gcc -c mainprog.c cel : zależności
24 Kiedy make wykona regułę ?1) Jeżeli plik celu nie istnieje 2) Jeżeli plik celu jest starszy niż któryś z plików określonych w zależnościach dla tego celu Makefile outprogname : binarytree.o mainprog.o gcc -o outprogname binarytree.o mainprog.o binarytree.o : binarytree.c binarytree.h gcc -c binarytree.c mainprog.o : mainprog.c binarytree.h gcc -c mainprog.c W zależnościach możemy podać pliki, które nie są jawnie wykorzystane w regule tworzenia, ale których zmiany powinny pociągnąć za sobą ponowne tworzenie celu.
25 Użycie zmiennych w pliku Makefilecc=gcc cflags=-O3 -g ldflags=-g outprogname : binarytree.o mainprog.o ${cc} ${ldflags} -o outprogname binarytree.o mainprog.o binarytree.o : binarytree.c ${cc} ${cflags} -c binarytree.c mainprog.o : mainprog.c ${cc} ${cflags} -c mainprog.c 1. 2. $ make gcc -O3 -g -c mainprog.c gcc -O3 -g -c binarytree.c gcc -g -o outprogname binarytree.o mainprog.o $ make cflags=-O2 gcc -O2 -c mainprog.c gcc -O2 -c binarytree.c gcc -g -o outprogname binarytree.o mainprog.o
26 Makefile - zmienne wbudowanecc=gcc cflags=-O3 -g ldflags=-g outprogname : binarytree.o mainprog.o ${cc} ${ldflags} -o $^ binarytree.o : binarytree.c ${cc} ${cflags} -c $< mainprog.o : mainprog.c - nazwa celu ($% dla archiwów) $< - pierwszy wymagany $^ - wszystkie wymagane, bez powtórzeń $+ - wszystkie wymagane, z powtórzeniami $? - wszystkie wymagane nowsze niż cel $ make gcc -O3 -g -c mainprog.c gcc -O3 -g -c binarytree.c gcc -g -o outprogname binarytree.o mainprog.o
27 Makefile - użycie dopasowywania wzorcówcc=gcc cflags=-O3 -g ldflags=-g outprogname : binarytree.o mainprog.o ${cc} ${ldflags} -o $^ %.o : %.c ${cc} ${cflags} -c $< Tworzenie wszystkich obiektów wymienionych we wcześniejszych zależnościach $ make gcc -O3 -g -c mainprog.c gcc -O3 -g -c binarytree.c gcc -g -o outprogname binarytree.o mainprog.o
28 Makefile - funkcje zaawansowaneFunkcje tekstowe: $(subst FROM,TO,TEXT) zamienia tekst FROM na tekst TO w tekście TEXT $(patsubst PATTERN,REPLACEMENT,TEXT) zamienianie ze wzorcem $(strip STRING) usuwa białe znaki z początku i końca $(filter PATTERN...,TEXT) zwraca te słowa w tekście które pasują do jednego ze wzorców $(filter-out PATTERN...,TEXT) odwrotnie niż poprzednia $(sort LIST) sortuje listę słów, usuwa duplikaty $(word N,TEXT) zwraca n-te słowo $(words TEXT) zwraca liczbę słów w tekście Funkcje od plików: $(dir NAZWY) wydobywa nazwy katalogów $(notdir NAZWY) wydobywa ostatni człon $(suffix NAZWY) wydobywa rozszerzenia $(basename NAZWY) wydobywa wszystko oprócz rozszerzeń wzór rozwinięcie $(patsubst %.c,%.o,x.c.c bar.c) x.c.o bar.o sources = foo.c bar.c baz.s ugh.h foo: $(sources) foo: foo.c bar.c baz.s ugh.h cc $(filter %.c %.s,$(sources)) -o foo cc foo.c bar.c baz.s -o foo
29 Przykład pliku MakefileCFLAGS+=-Wall -g CXXFLAGS+=-Wall -g MAIN_TARGETS_UNSORTED=$(patsubst %.cpp, %, $(wildcard *.cpp)) $(patsubst %.c, %,\ $(wildcard *.c)) MAIN_TARGETS=$(sort $(MAIN_TARGETS_UNSORTED)) all: $(MAIN_TARGETS) %: %.c %.h $(CC) $(CFLAGS) $(CPPFLAGS) $< -o %: %.c %: %.cpp %.h $(CXX) $(CXXFLAGS) $(CPPFLAGS) $< -o %: %.cpp clean: rm -f $(MAIN_TARGETS)
30 Debugowanie za pomocą GDBEdytowanie vi / vim Kompilowanie za pomocą GCC Wykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek
31 gdb - podstawy Uruchomienie gdbKompilacja programu - debugowany plik wykonywalny musi mieć dołączoną na etapie kompilacji i linkowania tablicę symboli (opcja -g) $ gcc -g prostokat.c ./prostokat $ gcc -g -c mainprog.c $ gcc -g -c binarytree.c $ gcc -g -o outprogname binarytree.o mainprog.o ./outprogname Uruchomienie gdb bez argumentów $ gdb z nazwą programu $ gdb outproganem z nazwą programu i pidem procesu $ gdb outproganem 12345 z nazwą programu i zrzutem pamięci $ gdb qsort2 core.2957 Core was generated by `qsort2'. Program terminated with signal 7, Emulator trap. #0 0x2734 in qsort2 (l=93643, u=93864, strat=1) at qsort2.c:118 do i++; while (i <= u && x[i] < t); (gdb) quit $ M.Loukides, A.Oram, Getting to Know gdb
32 gdb – pliki i uruchomienie programufile [file] użyj plik file jako plik wykonywalny i tablicę symboli core [file] użyj plik file jako zrzut pamięci load [file] dołącz dynamicznie plik file info share wyświetl nazwy wszystkich aktualnie załadowanych bibliotek dzielonych run arglist uruchom program z listą argumentów arglist run uruchom program z aktualną listą argumentów set args arglist ustal listę argumentów arglist dla kolejnego uruchomienia set args wyczyść listę argumentów dla kolejnego uruchomienia show args wyświetl listę argumentów dla kolejnego uruchomienia show env wyświetl wszystkie zmienne systemowe
33 gdb - pułapki (breakpoints)Użycie b jest równoważne użyciu break. b [file:]line ustaw breakpoint w linii line [w pliku file] b [file:]func ustaw breakpoint na funkcji func [w pliku file] b ... if expr przerwij warunkowo, jeżeli expr prawdziwe tbreak breakpoint tymczasowy (jednorazowy) rbreak regex przerywaj na wszystkich funkcjach dopasowanych do wyrażenia regex catch event przerwij po zdarzeniu event , (zdarzeniami mogą być catch, throw, exec, fork, vfork, load, unload) info break wyświetl zdefiniowane breakpointy clear usuń breakpoint z kolejnej instrukcji clear [file:]func usuń breakpoint z funkcji func clear [file:]line usuń breakpoint z linii line delete [n] usuń wszystkie breakpointy [lub o numerze n]
34 gdb - wykonanie programucontinue, c [count] kontynuuj wykonanie; jeżeli określono count to ignoruj count kolejnych breakpointów step, s [count] wykonaj pojedynczy krok (wchodzi do wnętrz funkcji); powtórz count razy next, n [count] wykonaj pojedynczy krok; powtórz count razy until location wykonaj aż do lokalizacji location (np. nr linii) finish wykonaj aż do zakończenia aktualnej f-cji i wyświetl zwróconą wartość jump line wznów wykonanie od linii line set var=expr zapisz do zmiennej o nazwie var wartość expr backtrace wyświetl wszystkie ramki ze stosu (m.in informacja o wywołanych funkcjach)
35 gdb - wyświetlanie danychprint, p [/f][expr] wyświetl wartość wyrażenia expr (albo wyrażenia ostatnio wyświetlanego) zgodnie z formatem f display [/f][expr] wyświetlaj wartość wyrażenia expr zawsze gdy program się zatrzyma (zgodnie z formatem f ) display wyświetl listę wszystkich obserwowanych zmiennych undisplay n usuń z listy obserwowanych zmiennych n -te wyrażenie
36 gdb - sygnały i przeglądanie koduhandle signal act ustal reakcję act gdb na sygnał signal; możliwe reakcje: print (wyświetl informację o pojawieniu się sygnału), noprint (nie wyświetlaj), stop (zatrzymaj wykonanie), nostop (nie zatrzymuj), pass (pozwól debugowanemu programowi na odebranie sygnału), nopass (nie przekazuj sygnału do programu) info signals wyświetl tablicę obsługiwanych sygnałów GDB list wyświetl 10 kolejnych linii kodu list - wyświetl 10 poprzednich linii kodu list [file:]num wyświetl 10 linii kodu wokół linii num w [pliku file] list [file:]func wyświetl 10 linii kodu wokół początku funkcji func w [pliku file] list f,l wyświetl kod od linii f do linii l info sources wyświetl listę wszystkich plików źródłowych
37 Interakcja programu ze środowiskiem wykonaniaEdytowanie vi / vim Kompilowanie za pomocą GCC Wykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek M.Mitchel, J.Oldham, A.Samuel, LINUX: Programowanie dla zaawansowanych
38 Argumenty wywołania i zmienne środowiskowe#include
39 stdin, stdout i stderr standardowy strumień wejścia: stdin 0 np. scanfstandardowy strumień wyjścia: stdout 1 np. printf standardowy strumień błędów: stderr 2 stdout jest buforowany, fflush( stdout) powoduje opróżnienie bufora f-cje read, write czy fprint, które oczekują jako argumentu uchwytu do pliku, mogą obsłużyć standardowe strumienie, np.: write( stderr, ”abc”, 4) przekierowanie wyjść do plików lub potoków przy wywołaniu programu (przekierowanie stderr do stdout musi być po przekierowaniu stdout do pliku, ale przed przekierowaniem do potoku): $ program > output_file.txt 2>&1 $ program 2>&1 | filter
40 Tworzenie i używanie bibliotekEdytowanie vi / vim Kompilowanie za pomocą GCC Wykorzystanie GNU Make Debugowanie za pomocą GDB Interakcja programu ze środowiskiem wykonania Tworzenie i używanie bibliotek
41 Tworzenie biblioteki statycznejBiblioteka statyczna jest plikiem archiwum (rozszerzenie *.a) zawierającym zbiór obiektów (rozszerzenie *.o), które powstały w wyniku kompilacji. Do tworzenia i zarządzania archiwami służy program ar. $ ar cr libname.a obj1.o obj2.o obj3.o ... create replace libtest.h int add( int a, int b); void shrink( char *str); libtest.c $ gcc -c libtest.c $ ar cr libtest.a libtest.o #include "libtest.h" int add( int a, int b) { return a+b;} void shrink( char *str) { str[1]=0;};
42 Użycie biblioteki statycznejAby dołączyć bibliotekę statyczną: dołączamy w kodzie źródłowym programu odniesienie do pliku nagłówkowego biblioteki podczas konsolidacji programu używamy przełącznika –l podając nazwę biblioteki (nazwa bez lib i .a) opcjonalnie wykorzystujemy przełącznik –L aby wskazać katalogi, które mogą zawierać bibliotekę Uwaga: kolejność wywołania przełączników ma znaczenie! Biblioteki należy dołączać na końcu linii poleceń, ponieważ linker przeszukuje je pod kątem wszystkich symboli, do których były odniesienia w przetworzonych wcześniej plikach i które nie zostały jeszcze zdefiniowane. Fragmenty kodu z bibliotek są kopiowane do pliku wynikowego. static.c #include "libtest.h" int main() { int c = add( 10, 20); return 0; } $ ls libtest.a static.c $ gcc static.c –o static –L. -ltest
43 Tworzenie biblioteki współdzielonejBiblioteka współdzielona jest zbiorem obiektów o rozszerzeniu *.so.#1.#2.#3 (numery #1, #2 i opcjonalne #3 oznaczają numery wersji – począwszy od najbardziej istotnego). Jest on linkowana przy pomocy gcc z opcjami –shared i –fPIC (opcja –fPIC również przy kompilacji obiektów). W kodzie biblioteki nie może być f-cji main, mogą być za to _init() i _fini()(po ustawieniu opcji -nostartfiles). $ gcc –shared -fPIC –o libname.so obj1.o obj2.o obj3.o ... libtest.h int add( int a, int b); void shrink( char *str); PIC (position independent code) libtest.c #include "libtest.h" #include
44 Użycie biblioteki współdzielonej - łączenieAby dołączyć bibliotekę współdzieloną podczas konsolidacji wykonujemy dokładnie te same czynności co przy dołączaniu biblioteki statycznej. Opcjonalnie możemy w kod wynikowy wkompilować informację o ścieżce, gdzie program będzie miał szukać bibliotekę podczas uruchamiania (opcja –Wl,-rpath,path). Kod programu nie zawiera kodu bibliotecznego, a jedynie odnośniki do niego. Bibliotek jest ładowana przy uruchamianiu programu. Każde podłączenie biblioteki powoduje wywołanie f-cji _init(), odłączenie - wywołanie f-cji _fini(). sshared.c #include "libtest.h" int main() { int c = add( 10, 20); return 0; } $ ls libtest.so sshared.c $ gcc sshared.c –o sshared –L. –ltest \ -Wl,-rpath,.
45 Użycie biblioteki współdzielonej – ładowanie (1)Bibliotek współdzielona może zostać załadowana dynamicznie podczas działania programu. Służą do tego f-cje biblioteki dl: 1) ładowanie biblioteki void *dlopen(const char *file, int mode); 2) obsługa błędów char *dlerror(void); 3) ładowanie symboli (m.in. funkcji) void *dlsym(void *restrict handle, const char *restrict name); 4) zwolnienei biblioteki int dlclose(void *handle); Funkcje wymagają dołączenia do programu pliku nagłókowego
46 Użycie biblioteki współdzielonej – ładowanie (2)dshared.c #include
47 Biblioteki - informacje dodatkoweBiblioteki podane przy konsolidacji po opcji –l wyszukiwane są standardowo w katalogach /lib i /usr/lib. Można to zmieniać za pomocą zmiennej środowiskowej LD_LIBRARY_PATH W systemie mogą istnieć wersje statyczna i współdzielona tej samej biblioteki: linker dołącza bibliotekę, na którą pierwszą się natknie przy przeszukiwaniu katalogów bibliotek; jeżeli w tym samym katalogu jest wersja statyczna i współdzielona, to pierwszeństwo ma wersja współdzielona; użycie opcji –static wymusza użycie wersji statycznej biblioteki; współdzielona wersja biblioteki sama rozwiązuje swoje zależności (czyli dołącza kolejne wymagane biblioteki); statyczna wersja biblioteki wymaga jawnego obsłużenia swoich zależności (przy linkowaniu programu trzeba dołączyć również biblioteki potrzebne dla bibliotek). Zależności w programie można sprawdzić za pomocą polecenia ldd Biblioteki funkcjonują pod 3 podstawowymi nazwami: real name, np. libtest.so.0.1 soname, np. libtest.so.0 ($ ln –s libtest.so.0.1 libtest.so.0) linker name, np. libtest.so ($ ln –s libtest.so.0.1 libtest.so)