Ristrutturazione della introduzione.
[gapil.git] / fileintro.tex
1 \chapter{I files: introduzione}
2 \label{cha:files_intro}
3  
4 Uno dei concetti fondamentali della architettura di unix è il cosiddetto
5 \textit{everything is a file}, cioè il fatto che l'accesso ai vari dispositivi
6 di input/output del computer viene effettuato attraverso un'interfaccia
7 astratta che tratta le periferiche allo stesso modo degli usuali file di
8 dati.
9
10 Questo significa che si può accedere cioè a qualunque periferica del computer,
11 dalla seriale, alla parallela, alla console, e agli stessi dischi attraverso i
12 cosiddetti file di dispositivo (i \textit{device files}). Questi sono dei file
13 speciali agendo sui quali i programmi possono leggere, scrivere e compiere
14 operazioni direttamente sulle periferiche, usando le stesse funzioni che si
15 usano per i normali file di dati.
16
17 In questo capitolo forniremo un'introduzione alle principali caratteristiche
18 di questa interfaccia, su come essa viene implementata in Linux e su come sono
19 organizzati i file nel sistema.
20
21
22 \section{I file in un sistema unix-like}
23 \label{sec:fileintr_overview}
24
25 Visto il ruolo fondamentale che i files vengono ad assumere in un sistema
26 unix, è anzitutto opportuno fornire un'introduzione dettagliata su come essi
27 vengono trattati dal sistema. In particolare occorre tenere presente dov'è che
28 si situa il limite fondamentale fra kernel space e user space che tracciavamo
29 al \capref{cha:intro_unix}.
30
31 Partiamo allora da come viene strutturata nel sistema la disposizione dei
32 file: per potervi accedere il kernel usa una apposita interfaccia che permetta
33 di strutturare l'informazione tenuta sullo spazio grezzo disponibile sui
34 dischi, cioè quello che si chiama un \textit{filesystem} (useremo per brevità
35 questo nome al posto della più prolissa traduzione italiana sistema di file). 
36
37 Sarà attraverso quest'ultimo che il kernel andrà a gestire l'accesso ai dati
38 memorizzati all'interno del disco stesso, strutturando l'informazione in files
39 e directory (su questo aspetto torneremo con maggiori dettagli in
40 \secref{sec:filedir_filesystem}).  Per poter accedere ai file contenuti in un
41 disco occorrerà perciò attivare il filesystem, questo viene fatto
42 \textsl{montando} il disco (o la partizione del disco).
43
44 %In generale un filesystem piazzerà opportunamente sul disco dei blocchi di
45 %informazioni riservate che tengono conto degli inodes allocati, di quelli
46 %liberi, e delle posizioni fisiche su disco dei dati contenuti nei files, per
47
48 In unix, a differenza di quanto avviene in altri sistemi operativi, tutti i
49 file vengono tenuti all'interno di un unico albero la cui radice (la directory
50 di \textit{root}) viene montata all'avvio. Pertanto un file viene identificato
51 dall'utente usando quello che viene chiamato \textit{pathname}, cioè il
52 percorso che si deve fare per accedere al file.
53
54 Dopo la fase di inizializzazione il kernel riceve dal boot loader
55 l'indicazione di quale dispositivo contiene il filesystem da usare come punto
56 di partenza e questo viene montato come radice dell'albero (cioè nella
57 directory \texttt{/}); tutti gli ulteriori dischi devono poi essere inseriti
58 nell'albero utilizzando opportune subdirectory.
59
60 Alcuni filesystem speciali (come \texttt{/proc} che contiene un'interfaccia ad
61 alcune strutture interne del kernel) sono generati automaticamente dal kernel
62 stesso, ma anche essi devono essere montati all'interno dell'albero.
63
64 All'interno dello stesso albero si potranno poi inserire anche gli altri
65 oggetti visti attraverso l'interfaccia che manipola i files come le FIFO, i
66 link, i socket e gli stessi i file di dispositivo (questi ultimi, per
67 convenzione, sono inseriti nella directory \texttt{/dev}).
68
69 \subsection{Il \textit{virtual filesystem} di Linux}
70 \label{sec:fileintr_vfs}
71
72 Esamineremo adesso come viene implementato l'accesso ai files in Linux. Questa
73 sezione riporta informazioni sui dettagli di come il kernel gestisce i files.
74 L'argomento è abbastanza ``esoterico'' e questa sezione può essere saltata ad
75 una prima lettura; è bene però tenere presente che vengono introdotti qui
76 alcuni termini che potranno comparire in seguito, come \textit{inode},
77 \textit{dentry}, \textit{dcache}.
78
79 In Linux il concetto di \textit{everything is a file} è stato implementato
80 attraverso il \textit{virtual filesystem} (da qui in avanti VFS) che è
81 l'interfaccia che il kernel rende disponibile ai programmi in user space
82 attraverso la quale vengono manipolati i files; esso provvede un livello di
83 indirezione che permette di collegare le operazioni di manipolazione sui files
84 alle operazioni di I/O e gestisce l'organizzazione di questi ultimi nei vari
85 modi in cui diversi filesystem la effettuano, permettendo la coesistenza
86 di filesystem differenti all'interno dello stesso albero delle directory
87
88 Quando un processo esegue una system call che opera su un file il kernel
89 chiama sempre una funzione implementata nel VFS; la funzione eseguirà le
90 manipolazioni sulle strutture generiche e ridirigendo la chiamata alla
91 opportune routine del filesystem specifico a cui si fa riferimento, saranno
92 queste a chiamare le funzioni di piu basso livello che eseguono le operazioni
93 di I/O sul dispositivo fisico, secondo lo schema riportato in \nfig.
94
95 \begin{figure}[htb]
96   \centering
97   
98   \caption{Schema delle operazioni del VFS}
99   \label{fig:fileintro_VFS_scheme}
100 \end{figure}
101
102 La funzione più importante implementata dal VFS è la system call \texttt{open}
103 che permette di aprire un file. Dato un pathname viene eseguita una ricerca
104 dentro la \textit{directory entry cache} (in breve \textit{dcache}),
105 una tabella di hash che contiene tutte le \textit{directory entry} (in breve
106 \textit{dentry}) che permette di associare in maniera rapida ed efficiente il
107 pathname a una specifica dentry.
108
109 Una singola dentry contiene in genere il puntatore ad un \textit{inode};
110 quest'ultimo è la struttura base che sta sul disco e che identifica un singolo
111 oggetto del VFS sia esso un file ordinario, una directory, una FIFO, un file
112 di dispositivo, o una qualsiasi altra cosa che possa essere rappresentata dal
113 VFS (sui tipi di ``files'' possibili torneremo in seguito). A ciascuno di essi
114 è associata pure una struttura che sta in memoria, e che oltre alle
115 informazioni sullo specifico file contiene pure il riferimento alle funzioni
116 (i \textsl{metodi}) da usare per poterlo manipolare.
117
118 Le dentries ``vivono'' in memoria e non vengono mai salvate su disco, vengono
119 usate per motivi di velocità, gli inodes invece stanno su disco e vengono
120 copiati in memoria quando serve, ed ogni cambiamento viene copiato
121 all'indietro sul disco, gli inodes che stanno in memoria sono inodes del VFS
122 ed è ad essi che puntano le singole dentry.
123
124 La dcache costituisce perciò una sorta di vista completa di tutto l'albero dei
125 files, ovviamente per non riempire tutta la memoria questa vista è parziale
126 (la dcache cioè contiene solo le dentry per i file per i quali è stato
127 richiesto l'accesso), quando si vuole risolvere un nuovo pathname il VFS deve
128 creare una nuova dentry e caricare l'inode corrispondente in memoria. 
129
130 Questo procedimento viene eseguito dal metodo \texttt{lookup()} dell'inode
131 della directory che contiene il file; questo viene installato nelle relative
132 strutture in memoria quando si effettua il montaggio lo specifico filesystem
133 su cui l'inode va a vivere.
134
135 Una volta che il VFS ha a disposizione la dentry (ed il relativo inode)
136 diventa possibile accedere alle varie operazioni sul file come la
137 \texttt{open} per aprire il file o la \texttt{stat} per leggere i dati
138 dell'inode e passarli in user space.
139
140 L'apertura di un file richiede comunque un'altra operazione, l'allocazione di
141 una struttura di tipo \texttt{file} in cui viene inserito un puntatore alla
142 dentry e una struttura \verb|f_ops| che contiene i puntatori ai metodi che
143 implementano le operazioni disponibili sul file. In questo modo i processi in
144 user space possono accedere alle operazioni attraverso detti metodi, che
145 saranno diversi a seconda del tipo di file (o dispositivo) aperto (su questo
146 torneremo in dettaglio in \secref{sec:fileunix_fd}). Un elenco delle operazioni
147 previste dal kernel è riportato in \ntab.
148
149 \begin{table}[htb]
150   \centering
151   \begin{tabular}[c]{c p{7cm}}
152     \textbf{funzione} & \textbf{operazione} \\
153     \hline
154     \textit{open}    & apre il file \\
155     \textit{read}    & legge dal file \\
156     \textit{write}   & scrive sul file \\ 
157     \textit{llseek}  & sposta la posizione corrente sul file \\
158     \textit{ioctl}   & accede alle operazioni di controllo 
159                        (tramite la \texttt{ioctl})\\
160     \textit{readdir} & per leggere il contenuto di una directory \\
161     \textit{poll}    & \\
162     \textit{mmap}    & chiamata dalla system call \texttt{mmap}. 
163                        mappa il file in memoria\\
164     \textit{release} & chiamata quando l'ultima referenza a un file 
165                        aperto è chiusa\\
166     \textit{fsync}   & chiamata dalla system call \texttt{fsync} \\
167     \textit{fasync}  & chiamate da \texttt{fcntl} quando è abilitato 
168                        il modo asincrono per l'I/O su file. \\
169     \hline
170   \end{tabular}
171   \caption{Operazioni sui file definite nel VFS.}
172   \label{tab:fileintr_file_operations}
173 \end{table}
174
175 In questo modo per ciascun file diventano utilizzabili una serie di operazioni
176 (non è dette che tutte siano disponibili), che costituiscono l'interfaccia
177 astratta del VFS, e qualora se ne voglia eseguire una il kernel andrà ad
178 utilizzare la opportuna routine dichiarata in \verb|f_ops| appropriata al tipo
179 di file in questione. 
180
181 Così sarà possibile scrivere sulla porta seriale come su un file di dati
182 normale; ovviamente certe operazioni (nel caso della seriale ad esempio la
183 \textit{seek}) non saranno disponibili, però con questo sistema l'utilizzo di
184 diversi filesystem (come quelli usati da Windows o MacOs) è immediato e
185 (relativamente) trasparente per l'utente ed il programmatore.
186
187 \subsection{Il controllo di accesso}
188 \label{sec:fileintr_access_ctrl}
189
190 In unix è implementata da qualunque filesystem standard una forma elementare
191 (ma adatta alla maggior parte delle esigenze) di controllo di accesso ai
192 files. Torneremo sull'argomento in dettaglio più avanti (vedi
193 \secref{sec:filedir_access_control}), qui ci limitiamo ad una introduzione dei
194 concetti essenziali.
195
196 Si tenga conto poi che quanto diremo è vero solo per filesystem di tipo Unix,
197 e non è detto che sia applicabile (ed infatti non è vero per il filesystem di
198 Windows) a un filesystem qualunque. Esistono inoltre estensioni che permettono
199 di implementare le ACL (\textit{Access Control List}) che sono un meccanismo
200 di controllo di accesso molto più sofisticato.
201
202 Ad ogni file Unix associa sempre l'utente che ne è proprietario (il cosiddetto
203 \textit{owner}) e il gruppo di appartenenza, secondo il meccanismo degli uid e
204 gid accennato in \secref{sec:intro_usergroup}, e un insieme di permessi che
205 sono divisi in tre classi, e cioè attribuiti rispettivamente al proprietario,
206 a qualunque utente faccia parte del gruppo cui appartiene il file, e a tutti
207 gli altri utenti.
208
209 I permessi sono espressi da un insieme di 12 bit: di questi i nove meno
210 significativi sono usati a gruppi di tre per indicare i permessi base di
211 lettura, scrittura ed esecuzione (indicati rispettivamente con le lettere
212 \textit{w}, \textit{r} \textit{x}) applicabili rispettivamente al
213 proprietario, al gruppo, a tutti (una descrizione più dettagliata dei vari
214 permessi associati ai file è riportata in \secref{sec:filedir_suid_sgid}).  I
215 restanti tre bit sono usati per indicare alcune caratteristiche più complesse
216 (\textit{suid}, \textit{sgid}, e \textit{sticky}) su cui pure torneremo in
217 seguito (vedi \secref{sec:filedir_suid_sgid} e \secref{sec:filedir_sticky}).
218
219 Tutte queste informazioni sono tenute per ciascun file nell'inode. Quando un
220 processo cerca l'accesso al file esso controlla i propri uid e gid
221 confrontandoli con quelli del file e se l'operazione richiesta è compatibile
222 con i permessi associati al file essa viene eseguita, altrimenti viene
223 bloccata ed è restituito un errore di \texttt{EPERM}. Questo procedimento non
224 viene eseguito per l'amministratore di sistema (il cui uid è zero) il quale ha
225 pertanto accesso senza restrizione a qualunque file del sistema.
226
227 In realtà il procedimento è più complesso di quanto descritto in maniera
228 elementare qui; inoltre ad un processo sono associati diversi identificatori,
229 torneremo su questo in maggiori dettagli in seguito in \secref{sec:proc_perms}.
230
231 \subsection{I tipi di files}
232 \label{sec:fileintr_file_types}
233
234 Come detto in precedenza esistono vari tipi di oggetti implementati del VFS
235 per i quali è disponibile l'interfaccia astratta da esso provveduta. Un elenco
236 dei vari tipi di file definiti nel VFS è il seguente:
237  
238 \begin{table}[htb]
239   \begin{center}
240     \begin{tabular}[c]{l l p{7cm}}
241     \multicolumn{2}{c}{\textbf{Nome}} & \textbf{Descrizione} \\
242     \hline
243       \textit{regular file} & \textsl{file normale} &
244       un file che contiene dei dati (l'accezione normale di file) \\
245       \textit{directory} & \textsl{cartella o direttorio} &
246       un file che contiene una lista di nomi associati a degli inodes \\
247       \textit{symbolic link} & \textsl{collegamento simbolico} &
248       un file che contiene un riferimento ad un altro file/directory \\
249       \textit{char device} & \textsl{dispositivo a caratteri} &
250       un file che identifica una periferica ad accesso sequenziale \\
251       \textit{block device} & \textsl{dispositivo a blocchi} &
252       un file che identifica una periferica ad accesso diretto \\
253       \textit{fifo} & \textsl{tubo} &
254       un file speciale che identifica una linea di comunicazione software
255       (unidirezionale) \\
256       \textit{socket} & \textsl{manicotto} &
257       un file speciale che identifica una linea di comunicazione software
258       (bidirezionale) \\
259     \hline
260     \end{tabular}
261     \caption{Tipologia dei file definiti nel VFS}
262     \label{tab:fileintr_file_types}
263   \end{center}
264 \end{table}
265
266 Tutto ciò non ha ovviamente nulla a che fare con la classificazione sui tipi
267 di file (in questo caso file di dati) in base al loro contenuto, o tipo di
268 accesso.  Una delle differenze principali con altri sistemi operativi (come il
269 VMS o Windows) è che per Unix tutti i file di dati sono identici e contengono
270 un flusso continuo di bytes; non esiste cioè differenza per come vengono visti
271 dal sistema file di diverso contenuto o formato (come nel caso di quella fra
272 file di testo e binari che c'è in Windows) né c'è una strutturazione a record
273 per il cosiddetto ``accesso diretto'' come nel caso del VMS.
274 %  (con i kernel
275 % della serie 2.4 è disponibile una forma di accesso diretto ai dischi il
276 % \textit{raw access} che però non ha nulla a che fare con questo).
277
278 Una seconda differenza è nel formato dei file ASCII; in Unix la fine riga è
279 codificata in maniera diversa da Windows o MacIntosh, in particolare il fine
280 riga è il carattere \texttt{LF} (o \verb|\n|) al posto del \texttt{CR}
281 (\verb|\r|) del mac e del \texttt{CR LF} di Windows. Questo può causare alcuni
282 problemi qualora nei programmi si facciano assunzioni sul terminatore della
283 riga.
284
285
286 \section{Una panoramica sull'uso dei file}
287 \label{sec:fileintr_io_overview}
288
289 Per poter accedere al contenuto dei file occorre anzitutto aprirlo. Questo
290 crea un canale di comunicazione che permette di eseguire una serie di
291 operazioni. Una volta terminate le operazioni, il file dovrà essere chiuso, e
292 questo chiuderà il canale di comunicazione impedendo ogni ulteriore
293 operazione.
294
295 \subsection{Le due interfacce ai file}
296 \label{sec:fileintr_io_api}
297
298 In unix le modalità di accesso ai file e le relative interfacce di
299 programmazione sono due, basate su due diversi meccanismi di connessione. 
300
301 La prima è l'interfaccia standard di unix, quella che il manuale delle glibc
302 chiama interfaccia dei descrittore di file (o \textit{file descriptor}).  È
303 un'interfaccia specifica di unix e provvede un accesso non bufferizzato.
304 L'interfaccia è primitiva ed essenziale, l'accesso viene detto non
305 bufferizzato in quanto la lettura e la scrittura vengono eseguite chiamando
306 direttamente le system call del kernel (in realtà il kernel effettua al suo
307 interno alcune bufferizzazioni per aumentare l'efficienza nell'accesso ai
308 dispositivi); i file descriptors sono rappresentati da numeri interi (cioè
309 semplici variabili di tipo \texttt{int}).  L'interfaccia è definita
310 nell'header \texttt{unistd.h}.
311
312 La seconda interfaccia è quella che il manuale della glibc chiama degli
313 \textit{stream}, essa provvede funzioni più evolute e un accesso bufferizzato
314 (controllato dalla implementazione fatta dalle librerie del C).  Questa è
315 l'interfaccia standard usata dal linguaggio C e perciò si trova anche su tutti
316 i sistemi non Unix. Gli stream sono oggetti complessi e sono rappresentati da
317 puntatori ad un opportuna struttura definita dalle librerie del C, si accede
318 ad essi sempre in maniera indiretta utilizzando il tipo \texttt{FILE *}.
319 L'interfaccia è definita nell'header \texttt{stdio.h}.
320
321 Entrambe le interfacce possono essere usate per l'accesso ai file come agli
322 altri oggetti del VFS (pipes, socket, device), ma per poter accedere alle
323 operazioni di controllo sul particolare tipo di oggetto del VFS scelto occorre
324 usare l'interfaccia standard di unix coi file descriptors. Allo stesso modo
325 devono essere usati i file descriptor se si vuole ricorrere a modalità
326 speciali di I/O come il polling o il non-bloccante (vedi
327 \secref{sec:file_xxx}).
328
329 Gli stream forniscono un'interfaccia di alto livello costruita sopra quella
330 dei file descriptor, che tratta tutti i file nello stesso modo, con
331 l'eccezione di poter scegliere tra diversi stili di bufferizzazione.  Il
332 maggior vantaggio degli stream è che l'interfaccia per le operazioni di
333 input/output è enormemente più ricca di quella dei file descriptor, che
334 provvedono solo funzioni elementari per la lettura/scrittura diretta di
335 blocchi di bytes.  In particolare gli stream dispongono di tutte le funzioni
336 di formattazione per l'input e l'output adatte per manipolare anche i dati in
337 forma di linee o singoli caratteri.
338
339 In ogni caso, dato che gli stream sono implementati sopra l'interfaccia
340 standard di unix, è sempre possibile estrarre il file descriptor da uno stream
341 ed eseguirvi operazioni di basso livello, o associare in un secondo tempo uno
342 stream ad un file descriptor.
343
344 In generale, se non necessitano specificatamente le funzionalità di basso
345 livello, è opportuno usare sempre gli stream per la loro maggiore portabilità
346 essendo questi ultimi definiti nello standard ANSI C; l'interfaccia con i file
347 descriptor invece segue solo lo standard POSIX.1 dei sistemi unix ed è
348 pertanto di portabilità più limitata.
349
350 \subsection{Caratteristiche specifiche dei file in unix}
351 \label{sec:fileint_unix_spec}
352
353 Essendo un sistema multitasking e multiutente esistono alcune caratteristiche
354 specifiche di Unix che devono essere tenute in conto nell'accesso ai file. È
355 infatti normale che più processi o programmi possano accedere
356 contemporaneamente allo stesso file e devono poter eseguire le loro operazioni
357 indipendentemente da quello che fanno gli altri processi.
358
359 Per questo motivo le strutture usate per all'accesso ai file sono relative al
360 processo che effettua l'accesso.  All'apertura di ogni file infatti viene
361 creata all'interno del processo una apposita struttura in cui sono memorizzati
362 tutti gli attributi del medesimo, che viene utilizzata per tutte le
363 operazioni. Questa è una struttura che resta locale al processo stesso; in
364 questo modo processi diversi possono usare le proprie strutture locali per
365 accedere ai file (che può essere sempre lo stesso) in maniera assolutamente
366 indipendente.
367
368 Questo ha delle conseguenze di cui è bene tenere conto; ad esempio in tutti i
369 sistemi POSIX uno degli attributi di un file aperto è la posizione corrente nel
370 file, cioè il punto nel file in cui verrebbe letto o scritto alla operazione
371 successiva. Essa è rappresentata da un numero intero che indica il numero di
372 bytes dall'inizio del file, che viene (a meno che non si apra il file in
373 append) inizializzato a zero all'apertura del medesimo.
374
375 Questo è uno dei dati che viene mantenuto nella suddetta struttura, per cui
376 ogni processo avrà la sua posizione corrente nel file, che non sarà
377 influenzata da quello che altri processi possono fare. Anzi, aprire un file
378 significa appunto creare ed inizializzare una tale struttura, per cui se si
379 apre due volte lo stesso file all'interno dello stesso processo, si otterranno
380 due file descriptor o due stream che avranno ancora una posizione corrente nel
381 file assolutamente indipendente.
382
383 Si tenga conto inoltre che un'altro dei dati contenuti nella struttura di
384 accesso è un riferimento all'inode del file, pertanto anche se il file viene
385 cancellato da un altro processo, sarà sempre possibile mantenere l'accesso ai
386 dati, e lo spazio su disco non verrà rilasciato fintanto che il file non sarà
387 chiuso e l'ultimo riferimento cancellato. È pertanto possibile (come vedremo
388 in dettaglio in \secref{sec:filedir_link}) aprire un file provvisorio per
389 cancellarlo immediatamente dopo; in questo modo all'uscita del programma il
390 file scomparirà definitivamente dal disco, ma il file ed il suo contenuto
391 saranno disponibili per tutto il tempo in cui il processo è attivo.
392
393 Ritorneremo su questo più avanti, quando tratteremo l'input/output sui file,
394 esaminando in dettaglio come tutto ciò viene realizzato.
395
396 Si ricordi infine che in unix non esistono i tipi di file e che non c'è nessun
397 supporto per le estensioni come parte del filesystem. Ciò non ostante molti
398 programmi adottano delle convenzioni per i nomi dei file, ad esempio il codice
399 C normalmente si mette in file con l'estensione .c, ma questa è, appunto, solo
400 una convenzione.
401