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