Correzioni di claudio
[gapil.git] / fileintro.tex
1 \chapter{L'architettura dei file}
2 \label{cha:file_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 dati.
8
9 Questo significa che si può accedere a qualunque periferica del computer,
10 dalla seriale, alla parallela, alla console, e agli stessi dischi attraverso i
11 cosiddetti file di dispositivo (i \textit{device file}). Questi sono dei file
12 speciali agendo sui quali i programmi possono leggere, scrivere e compiere
13 operazioni direttamente sulle periferiche, usando le stesse funzioni che si
14 usano per i normali file di dati.
15
16 In questo capitolo forniremo un'introduzione all'architettura della gestione
17 dei file, sia nelle sue caratteristiche generiche comuni a tutti gli unix, che
18 nelle particolarità che ha la specifica implementazione usata da Linux. Al
19 contempo tratteremo l'organizzazione dei file in un sistema unix-like, e le
20 varie caratteristiche distintive.
21
22
23
24 \section{L'organizzazione di file e directory}
25 \label{sec:file_organization}
26
27 Il primo passo nella trattazione dell'architettura della gestione dei file in
28 un sistema unix-like, è quello dell'esame di come essi vengono organizzati e
29 di quale è la struttura che hanno all'interno del sistema.
30
31
32 \subsection{La struttura di file e directory}
33 \label{sec:file_file_struct}
34
35 Partiamo allora da come viene strutturata nel sistema la disposizione dei
36 file: per potervi accedere il kernel usa una apposita interfaccia che permetta
37 di accedere all'informazione tenuta sullo spazio grezzo disponibile sui
38 dischi, cioè quello che si chiama un \textit{filesystem}\footnote{useremo per
39   brevità questo nome al posto della più prolissa traduzione italiana sistema
40   di file}, che descriveremo in dettaglio in \secref{sec:file_vfs}.
41
42 Sarà attraverso quest'ultimo che il kernel andrà a gestire l'accesso ai dati
43 memorizzati all'interno del disco stesso, strutturando l'informazione in file
44 e directory.  Per poter accedere ai file contenuti in un disco occorrerà
45 perciò attivare il filesystem, questo viene fatto \textsl{montando} il disco
46 (o la partizione del disco).
47
48 %In generale un filesystem piazzerà opportunamente sul disco dei blocchi di
49 %informazioni riservate che tengono conto degli inodes allocati, di quelli
50 %liberi, e delle posizioni fisiche su disco dei dati contenuti nei file, per
51
52 In unix, a differenza di quanto avviene in altri sistemi operativi, tutti i
53 file vengono tenuti all'interno di un unico albero la cui radice (la directory
54 di \textit{root}) viene montata all'avvio. Pertanto un file viene identificato
55 dall'utente usando quello che viene chiamato \textit{pathname}, cioè il
56 percorso che si deve fare per accedere al file.
57
58 Dopo la fase di inizializzazione il kernel riceve dal boot loader
59 l'indicazione di quale dispositivo contiene il filesystem da usare come punto
60 di partenza e questo viene montato come radice dell'albero (cioè nella
61 directory \file{/}); tutti gli ulteriori dischi devono poi essere inseriti
62 nell'albero utilizzando opportune subdirectory.
63
64 Alcuni filesystem speciali (come \file{/proc} che contiene un'interfaccia ad
65 alcune strutture interne del kernel) sono generati automaticamente dal kernel
66 stesso, ma anche essi devono essere montati all'interno dell'albero.
67
68 All'interno dello stesso albero si potranno poi inserire anche gli altri
69 oggetti visti attraverso l'interfaccia che manipola i file come le FIFO, i
70 link, i socket e gli stessi i file di dispositivo (questi ultimi, per
71 convenzione, sono inseriti nella directory \file{/dev}).
72
73 L'organizzazione dei nomi dei file deriva direttamente dall'organizzazione dei
74 medesimi nell'albero descritto in precedenza; una directory comunque, come già
75 specificato in \secref{sec:file_vfs}, è solo un particolare tipo di file
76 che contiene le informazioni che associano un nome al contenuto.
77
78 % Per questo, anche se è usuale parlare di ``file in una directory'' in realtà
79 % una directory contiene solo delle etichette per fare riferimento ai file
80 % stessi.
81
82 I manuale delle glibc chiama i nomi contenuti nelle directory
83 \textsl{componenti} (in inglese \textit{file name components}), noi li
84 chiameremo più semplicemente \textit{nomi}. Un file può essere indicato
85 rispetto alla directory corrente semplicemente specificando il nome da essa
86 contenuto. Una directory contiene semplicemente un elenco di questi nomi, che
87 possono corrispondere a un qualunque oggetto del filesystem, compresa un'altra
88 directory; l'albero viene appunto creato inserendo directory in altre
89 directory.
90
91 Il nome completo di file generico è composto da una serie di nomi separati da
92 una \file{/} (in Linux più \file{/} consecutive sono considerate
93 equivalenti ad una sola). Il nome completo di un file viene usualmente
94 chiamato \textit{pathname}, e anche se il manuale della glibc depreca questo
95 nome (poiché genererebbe confusione, dato che con \textit{path} si indica
96 anche un insieme di directory su cui effettuare una ricerca, come quello in
97 cui si cercano i comandi); non seguiremo questa scelta dato che l'uso della
98 parola \textit{pathname} è ormai così comune che è senz'altro più chiaro
99 dell'alternativa proposta.
100
101 Il processo con cui si associa ad un pathname uno specifico file è chiamato
102 risoluzione del nome (\textit{file name resolution} o \textit{pathname
103   resolution}).  La risoluzione viene fatta esaminando il pathname da destra a
104 sinistra e localizzando ogni nome nella directory indicata dal nome
105 precedente: ovviamente perché il procedimento funzioni occorre che i nomi
106 indicati come directory esistano e siano effettivamente directory, inoltre i
107 permessi devono consentire l'accesso.
108
109 Se il pathname comincia per \file{/} la ricerca parte dalla directory radice
110 del processo; questa, a meno di un \textit{chroot} (su cui torneremo in
111 seguito, vedi \secref{sec:xxx_chroot}) è la stessa per tutti i processi ed
112 equivale alla directory radice dell'albero (come descritto in
113 \secref{sec:file_organization}): in questo caso si parla di un pathname
114 \textsl{assoluto}. Altrimenti la ricerca parte dalla directory corrente (su
115 cui torneremo più avanti in \secref{sec:file_work_dir}) ed il pathname è
116 detto \textsl{relativo}.
117
118 I nomi \file{.} e \file{..} hanno un significato speciale e vengono inseriti
119 in ogni directory, il primo fa riferimento alla directory corrente e il
120 secondo alla directory \textsl{genitrice} (\textit{parent directory}) cioè la
121 directory che contiene il riferimento alla directory corrente; nel caso questa
122 sia la directory radice allora il riferimento è a se stessa.
123
124
125 \subsection{I tipi di file}
126 \label{sec:file_file_types}
127
128 Come detto in precedenza in unix esistono vari tipi di file, in Linux questi
129 sono implementati come oggetti del \textit{Virtual File System} (vedi
130 \secref{sec:file_vfs}) e sono presenti in tutti i filesystem unix-like
131 utilizzabili con Linux. L'elenco dei vari tipi di file definiti dal Virtual
132 File System è riportato in \ntab.
133
134 Si tenga ben presente che questa classificazione non ha nulla a che fare con
135 la classificazione sui tipi di file (che in questo caso sono sempre file di
136 dati) in base al loro contenuto, o tipo di accesso.
137
138 \begin{table}[htb]
139   \footnotesize
140   \centering
141     \begin{tabular}[c]{|l|l|p{7cm}|}
142     \hline
143     \multicolumn{2}{|c|}{\textbf{Tipo di file}} & \textbf{Descrizione} \\
144     \hline
145     \hline
146       \textit{regular file} & \textsl{file normale} &
147       un file che contiene dei dati (l'accezione normale di file) \\
148       \textit{directory} & \textsl{cartella o direttorio} &
149       un file che contiene una lista di nomi associati a degli inodes \\
150       \textit{symbolic link} & \textsl{collegamento simbolico} &
151       un file che contiene un riferimento ad un altro file/directory \\
152       \textit{char device} & \textsl{dispositivo a caratteri} &
153       un file che identifica una periferica ad accesso sequenziale \\
154       \textit{block device} & \textsl{dispositivo a blocchi} &
155       un file che identifica una periferica ad accesso diretto \\
156       \textit{fifo} & \textsl{tubo} &
157       un file speciale che identifica una linea di comunicazione software
158       (unidirezionale) \\
159       \textit{socket} & \textsl{manicotto} &
160       un file speciale che identifica una linea di comunicazione software
161       (bidirezionale) \\
162     \hline
163     \end{tabular}
164     \caption{Tipologia dei file definiti nel VFS}
165     \label{tab:file_file_types}
166 \end{table}
167
168 Infatti una delle differenze principali con altri sistemi operativi (come il
169 VMS o Windows) è che per Unix tutti i file di dati sono identici e contengono
170 un flusso continuo di byte. Non esiste cioè differenza per come vengono visti
171 dal sistema file di diverso contenuto o formato (come nel caso di quella fra
172 file di testo e binari che c'è in Windows) né c'è una strutturazione a record
173 per il cosiddetto ``accesso diretto'' come nel caso del VMS\footnote{con i
174   kernel della serie 2.4 è disponibile una forma di accesso diretto ai dischi
175   (il \textit{raw access}) attraverso dei device file appositi, che però non
176   ha nulla a che fare con questo}.
177
178 Una seconda differenza è nel formato dei file ASCII; in Unix la fine riga è
179 codificata in maniera diversa da Windows o Mac, in particolare il fine
180 riga è il carattere \texttt{LF} (o \verb|\n|) al posto del \texttt{CR}
181 (\verb|\r|) del Mac e del \texttt{CR LF} di Windows. Questo può causare alcuni
182 problemi qualora nei programmi si facciano assunzioni sul terminatore della
183 riga.
184
185
186 \subsection{Le due interfacce ai file}
187 \label{sec:file_io_api}
188
189 In unix le modalità di accesso ai file e le relative interfacce di
190 programmazione sono due, basate su due diversi meccanismi con cui è possibile
191 accedere al loro contenuto.
192
193 La prima è l'interfaccia standard di unix, quella che il manuale delle glibc
194 chiama interfaccia dei descrittori di file (o \textit{file descriptor}).  È
195 un'interfaccia specifica di unix e provvede un accesso non bufferizzato, la
196 tratteremo in dettaglio in \capref{cha:file_unix_interface}.
197
198 L'interfaccia è primitiva ed essenziale, l'accesso viene detto non
199 bufferizzato in quanto la lettura e la scrittura vengono eseguite chiamando
200 direttamente le system call del kernel (in realtà il kernel effettua al suo
201 interno alcune bufferizzazioni per aumentare l'efficienza nell'accesso ai
202 dispositivi); i file descriptor sono rappresentati da numeri interi (cioè
203 semplici variabili di tipo \type{int}).  L'interfaccia è definita
204 nell'header \file{unistd.h}.
205
206 La seconda interfaccia è quella che il manuale della glibc chiama degli
207 \textit{stream}, essa provvede funzioni più evolute e un accesso bufferizzato
208 (controllato dalla implementazione fatta dalle librerie del C), la tratteremo
209 in dettaglio in \capref{cha:files_std_interface}.
210
211 Questa è l'interfaccia standard specificata dall'ANSI C e perciò si trova
212 anche su tutti i sistemi non unix. Gli stream sono oggetti complessi e sono
213 rappresentati da puntatori ad un opportuna struttura definita dalle librerie
214 del C, si accede ad essi sempre in maniera indiretta utilizzando il tipo
215 \type{FILE *}.  L'interfaccia è definita nell'header \type{stdio.h}.
216
217 Entrambe le interfacce possono essere usate per l'accesso ai file come agli
218 altri oggetti del VFS (pipe, socket, device), ma per poter accedere alle
219 operazioni di controllo sul particolare tipo di oggetto del VFS scelto occorre
220 usare l'interfaccia standard di unix coi file descriptor. Allo stesso modo
221 devono essere usati i file descriptor se si vuole ricorrere a modalità
222 speciali di I/O come il polling o il non-bloccante (vedi
223 \secref{sec:file_noblocking}).
224
225 Gli stream forniscono un'interfaccia di alto livello costruita sopra quella
226 dei file descriptor, che tratta tutti i file nello stesso modo, con
227 l'eccezione di poter scegliere tra diversi stili di bufferizzazione.  Il
228 maggior vantaggio degli stream è che l'interfaccia per le operazioni di
229 input/output è enormemente più ricca di quella dei file descriptor, che
230 provvedono solo funzioni elementari per la lettura/scrittura diretta di
231 blocchi di byte.  In particolare gli stream dispongono di tutte le funzioni
232 di formattazione per l'input e l'output adatte per manipolare anche i dati in
233 forma di linee o singoli caratteri.
234
235 In ogni caso, dato che gli stream sono implementati sopra l'interfaccia
236 standard di unix, è sempre possibile estrarre il file descriptor da uno stream
237 ed eseguirvi operazioni di basso livello, o associare in un secondo tempo uno
238 stream ad un file descriptor.
239
240 In generale, se non necessitano specificatamente le funzionalità di basso
241 livello, è opportuno usare sempre gli stream per la loro maggiore portabilità
242 essendo questi ultimi definiti nello standard ANSI C; l'interfaccia con i file
243 descriptor invece segue solo lo standard POSIX.1 dei sistemi unix ed è
244 pertanto di portabilità più limitata.
245
246
247 \subsection{Caratteristiche specifiche dei file in unix}
248 \label{sec:fileint_unix_spec}
249
250 Essendo un sistema multitasking e multiutente esistono alcune caratteristiche
251 specifiche di un sistema unix-like che devono essere tenute in conto
252 nell'accesso ai file. È infatti normale che più processi o programmi possano
253 accedere contemporaneamente allo stesso file e devono poter eseguire le loro
254 operazioni indipendentemente da quello che fanno gli altri processi.
255
256 Per questo motivo le strutture usate per all'accesso ai file sono relative al
257 processo che effettua l'accesso.  All'apertura di ogni file infatti viene
258 creata all'interno del processo una apposita struttura in cui sono memorizzati
259 tutti gli attributi del medesimo, che viene utilizzata per tutte le
260 operazioni. Questa è una struttura che resta locale al processo stesso; in
261 questo modo processi diversi possono usare le proprie strutture locali per
262 accedere ai file (che può essere sempre lo stesso) in maniera assolutamente
263 indipendente.
264
265 Questo ha delle conseguenze di cui è bene tenere conto; ad esempio in tutti i
266 sistemi POSIX uno degli attributi di un file aperto è la posizione corrente nel
267 file, cioè il punto nel file in cui verrebbe letto o scritto alla operazione
268 successiva. Essa è rappresentata da un numero intero che indica il numero di
269 byte dall'inizio del file, che viene (a meno che non si apra il file in
270 append) inizializzato a zero all'apertura del medesimo.
271
272 Questo è uno dei dati che viene mantenuto nella suddetta struttura, per cui
273 ogni processo avrà la sua posizione corrente nel file, che non sarà
274 influenzata da quello che altri processi possono fare. Anzi, aprire un file
275 significa appunto creare ed inizializzare una tale struttura, per cui se si
276 apre due volte lo stesso file all'interno dello stesso processo, si otterranno
277 due file descriptor o due stream che avranno ancora una posizione corrente nel
278 file assolutamente indipendente.
279
280 Si tenga conto inoltre che un'altro dei dati contenuti nella struttura di
281 accesso è un riferimento all'inode del file, pertanto anche se il file viene
282 cancellato da un altro processo, sarà sempre possibile mantenere l'accesso ai
283 dati, e lo spazio su disco non verrà rilasciato fintanto che il file non sarà
284 chiuso e l'ultimo riferimento cancellato. È pertanto possibile (come vedremo
285 in dettaglio in \secref{sec:file_link}) aprire un file provvisorio per
286 cancellarlo immediatamente dopo; in questo modo all'uscita del programma il
287 file scomparirà definitivamente dal disco, ma il file ed il suo contenuto
288 saranno disponibili per tutto il tempo in cui il processo è attivo.
289
290 Ritorneremo su questo più avanti, quando tratteremo l'input/output sui file,
291 esaminando in dettaglio come tutto ciò viene realizzato.
292
293 Si ricordi infine che in ambiente unix non esistono i tipi di file e che non
294 c'è nessun supporto per le estensioni come parte del filesystem. Ciò non
295 ostante molti programmi adottano delle convenzioni per i nomi dei file, ad
296 esempio il codice C normalmente si mette in file con l'estensione .c, ma
297 questa è, appunto, solo una convenzione.
298
299
300 \section{L'architettura della gestione dei file}
301 \label{sec:file_architecture}
302
303 Per capire fino in fondo le proprietà di file e directory in un sistema
304 unix-like ed il funzionamento delle relative funzioni di manipolazione occorre
305 una breve introduzione sulla gestione dei medesimo e sugli oggetti su cui è
306 basato un filesystem di tipo unix. In particolare occorre tenere presente
307 dov'è che si situa la divisione fondamentale fra kernel space e user space che
308 tracciavamo al \capref{cha:intro_unix}.
309
310 In questa sezione esamineremo come viene implementato l'accesso ai file in
311 Linux, come il kernel può gestire diversi tipi di filesystem, descrivendo
312 poi in maniera un po' più dettagliata il filesystem standard di Linux,
313 l'\acr{ext2}, come esempio di un filesystem unix-like.
314
315
316 % in particolare si riprenderà, approfondendolo sul piano
317 % dell'uso nelle funzioni di libreria, il concetto di \textit{inode} di cui
318 % abbiamo brevemente accennato le caratteristiche (dal lato dell'implementazione
319 % nel kernel) in \secref{sec:file_vfs}.
320
321
322 \subsection{Il \textit{virtual filesystem} di Linux}
323 \label{sec:file_vfs}
324
325 % Questa sezione riporta informazioni sui dettagli di come il kernel gestisce i
326 % file.  L'argomento è abbastanza ``esoterico'' e questa sezione può essere
327 % saltata ad una prima lettura; è bene però tenere presente che vengono
328 % introdotti qui alcuni termini che potranno comparire in seguito, come
329 % \textit{inode}, \textit{dentry}, \textit{dcache}.
330
331 In Linux il concetto di \textit{everything is a file} è stato implementato
332 attraverso il \textit{Virtual File System} (da qui in avanti VFS) che è
333 l'interfaccia che il kernel rende disponibile ai programmi in user space
334 attraverso la quale vengono manipolati i file; esso provvede un livello di
335 indirezione che permette di collegare le operazioni di manipolazione sui file
336 alle operazioni di I/O e gestisce l'organizzazione di questi ultimi nei vari
337 modi in cui diversi filesystem la effettuano, permettendo la coesistenza
338 di filesystem differenti all'interno dello stesso albero delle directory
339
340 Quando un processo esegue una system call che opera su un file il kernel
341 chiama sempre una funzione implementata nel VFS; la funzione eseguirà le
342 manipolazioni sulle strutture generiche e utilizzerà poi la chiamata alla
343 opportune routine del filesystem specifico a cui si fa riferimento. Saranno
344 queste a chiamare le funzioni di più basso livello che eseguono le operazioni
345 di I/O sul dispositivo fisico, secondo lo schema riportato in \nfig.
346
347 \begin{figure}[htb]
348   \centering
349   \includegraphics[width=7cm]{img/vfs}
350   \caption{Schema delle operazioni del VFS}
351   \label{fig:file_VFS_scheme}
352 \end{figure}
353
354 Il VFS definisce un insieme di funzioni che tutti i filesystem devono
355 implementare. L'interfaccia comprende tutte le funzioni che riguardano i
356 file; le operazioni sono suddivise su tre tipi di oggetti: filesystem, inode
357 e file, corrispondenti a tre apposite strutture definite nel kernel.
358
359 Il VFS usa una tabella mantenuta dal kernel che contiene il nome di ciascun
360 filesystem supportato: quando si vuole inserire il supporto di un nuovo
361 filesystem tutto quello che occorre è chiamare la funzione
362 \func{register\_filesystem} passandole un'apposita struttura
363 (\var{file\_system\_type}) che contiene i dettagli per il riferimento
364 all'implementazione del medesimo, che sarà aggiunta alla citata tabella.
365
366
367 In questo modo quando viene effettuata la richiesta di montare un nuovo disco
368 (o qualunque altro \textit{block device} che può contenere un filesystem), il
369 VFS può ricavare dalla citata tabella il puntatore alle funzioni da chiamare
370 nelle operazioni di montaggio. Quest'ultima è responsabile di leggere da disco
371 il superblock (vedi \ref{sec:file_ext2}), inizializzare tutte le
372 variabili interne e restituire uno speciale descrittore dei filesystem montati
373 al VFS; attraverso quest'ultimo diventa possibile accedere alle routine
374 specifiche per l'uso di quel filesystem.
375
376 Il primo oggetto usato dal VFS è il descrittore di filesystem, un puntatore ad
377 una apposita struttura che contiene vari dati come le informazioni comuni ad
378 ogni filesystem, i dati privati relativi a quel filesystem specifico, e i
379 puntatori alle funzioni del kernel relative al filesystem. Il VFS può così
380 usare le funzioni contenute nel filesystem descriptor per accedere alle routine
381 specifiche di quel filesystem.
382
383 Gli altri due descrittori usati dal VFS sono relativi agli altri due oggetti
384 su cui è strutturata l'interfaccia. Ciascuno di essi contiene le informazioni
385 relative al file in uso, insieme ai puntatori alle funzioni dello specifico
386 filesystem usate per l'accesso dal VFS; in particolare il descrittore
387 dell'inode contiene i puntatori alle funzioni che possono essere usate su
388 qualunque file (come \func{link}, \func{stat} e \func{open}), mentre il
389 descrittore di file contiene i puntatori alle funzioni che vengono usate sui
390 file già aperti.
391
392
393 \subsection{Il funzionamento del VFS}
394 \label{sec:file_vfs_work}
395
396 La funzione più fondamentale implementata dal VFS è la system call
397 \func{open} che permette di aprire un file. Dato un pathname viene eseguita
398 una ricerca dentro la \textit{directory entry cache} (in breve
399 \textit{dcache}), una tabella di hash che contiene tutte le \textit{directory
400   entry} (in breve \textit{dentry}) che permette di associare in maniera
401 rapida ed efficiente il pathname a una specifica dentry.
402
403 Una singola \textit{dentry} contiene in genere il puntatore ad un
404 \textit{inode}; quest'ultimo è la struttura base che sta sul disco e che
405 identifica un singolo oggetto del VFS sia esso un file ordinario, una
406 directory, un link simbolico, una FIFO, un file di dispositivo, o una
407 qualsiasi altra cosa che possa essere rappresentata dal VFS (sui tipi di
408 ``file'' possibili torneremo in seguito). A ciascuno di essi è associata pure
409 una struttura che sta in memoria, e che oltre alle informazioni sullo
410 specifico file contiene pure il riferimento alle funzioni (i \textsl{metodi})
411 da usare per poterlo manipolare.
412
413 Le \textit{dentry} ``vivono'' in memoria e non vengono mai salvate su disco,
414 vengono usate per motivi di velocità, gli inode invece stanno su disco e
415 vengono copiati in memoria quando serve, ed ogni cambiamento viene copiato
416 all'indietro sul disco, gli inode che stanno in memoria sono inode del VFS ed
417 è ad essi che puntano le singole \textit{dentry}.
418
419 La \textit{dcache} costituisce perciò una sorta di vista completa di tutto
420 l'albero dei file, ovviamente per non riempire tutta la memoria questa vista è
421 parziale (la \textit{dcache} cioè contiene solo le \textit{dentry} per i file
422 per i quali è stato richiesto l'accesso), quando si vuole risolvere un nuovo
423 pathname il VFS deve creare una nuova \textit{dentry} e caricare l'inode
424 corrispondente in memoria.
425
426 Questo procedimento viene eseguito dal metodo \func{lookup()} dell'inode
427 della directory che contiene il file; questo viene installato nelle relative
428 strutture in memoria quando si effettua il montaggio lo specifico filesystem
429 su cui l'inode va a vivere.
430
431 Una volta che il VFS ha a disposizione la dentry (ed il relativo inode)
432 diventa possibile accedere alle varie operazioni sul file come la
433 \func{open} per aprire il file o la \func{stat} per leggere i dati
434 dell'inode e passarli in user space.
435
436 L'apertura di un file richiede comunque un'altra operazione, l'allocazione di
437 una struttura di tipo \var{file} in cui viene inserito un puntatore alla
438 \textit{dentry} e una struttura \var{f\_ops} che contiene i puntatori ai
439 metodi che implementano le operazioni disponibili sul file. In questo modo i
440 processi in user space possono accedere alle operazioni attraverso detti
441 metodi, che saranno diversi a seconda del tipo di file (o dispositivo) aperto
442 (su questo torneremo in dettaglio in \secref{sec:file_fd}). Un elenco delle
443 operazioni previste dal kernel è riportato in \ntab.
444
445 \begin{table}[htb]
446   \centering
447   \footnotesize
448   \begin{tabular}[c]{|l|p{7cm}|}
449     \hline
450     \textbf{Funzione} & \textbf{Operazione} \\
451     \hline
452     \hline
453     \textsl{\func{open}}   & apre il file \\
454     \textsl{\func{read}}   & legge dal file \\
455     \textsl{\func{write}}  & scrive sul file \\ 
456     \textsl{\func{llseek}} & sposta la posizione corrente sul file \\
457     \textsl{\func{ioctl}}  & accede alle operazioni di controllo 
458                        (tramite la \func{ioctl})\\
459     \textsl{\func{readdir}}& per leggere il contenuto di una directory \\
460     \textsl{\func{poll}}   & \\
461     \textsl{\func{mmap}}   & chiamata dalla system call \func{mmap}. 
462                        mappa il file in memoria\\
463     \textsl{\func{release}}& chiamata quando l'ultima referenza a un file 
464                        aperto è chiusa\\
465     \textsl{\func{fsync}}  & chiamata dalla system call \func{fsync} \\
466     \textsl{\func{fasync}} & chiamate da \func{fcntl} quando è abilitato 
467                            il modo asincrono per l'I/O su file. \\
468     \hline
469   \end{tabular}
470   \caption{Operazioni sui file definite nel VFS.}
471   \label{tab:file_file_operations}
472 \end{table}
473
474 In questo modo per ciascun file diventano utilizzabili una serie di operazioni
475 (non è dette che tutte siano disponibili), che costituiscono l'interfaccia
476 astratta del VFS, e qualora se ne voglia eseguire una il kernel andrà ad
477 utilizzare la opportuna routine dichiarata in \var{f\_ops} appropriata al tipo
478 di file in questione. 
479
480 Così sarà possibile scrivere sulla porta seriale come su un file di dati
481 normale; ovviamente certe operazioni (nel caso della seriale ad esempio la
482 \func{seek}) non saranno disponibili, però con questo sistema l'utilizzo di
483 diversi filesystem (come quelli usati da Windows o MacOs) è immediato e
484 (relativamente) trasparente per l'utente ed il programmatore.
485
486
487 \subsection{Il funzionamento di un filesystem unix}
488 \label{sec:file_filesystem}
489
490 Come già accennato in \secref{sec:file_organization} Linux (ed ogni unix
491 in generale) organizza i dati che tiene su disco attraverso l'uso di un
492 filesystem. Una delle caratteristiche di Linux rispetto agli altri unix è
493 quella di poter supportare grazie al VFS una enorme quantità di filesystem
494 diversi, ognuno dei quali ha una sua particolare struttura e funzionalità
495 proprie; per questo non entreremo nei dettagli di un filesystem specifico, ma
496 daremo una descrizione a grandi linee che si adatta alle caratteristiche
497 comuni di un qualunque filesystem standard unix.
498
499 Dato un disco lo spazio fisico viene usualmente diviso in partizioni; ogni
500 partizione può contenere un filesystem; la strutturazione tipica
501 dell'informazione su un disco è riportata in \nfig; in essa si fa riferimento
502 alla struttura del filesystem \acr{ext2}, che prevede una separazione dei dati
503 in \textit{blocks group} che replicano il superblock (ma sulle caratteristiche
504 di \acr{ext2} torneremo in \secref{sec:file_ext2}). È comunque caratteristica
505 comune di tutti i filesystem unix, indipendentemente da come poi viene
506 strutturata nei dettagli questa informazione, prevedere una divisione fra la
507 lista degli inodes e lo spazio a disposizione per i dati e le directory.
508
509 \begin{figure}[htb]
510   \centering
511   \includegraphics[width=9cm]{img/disk_struct}
512   \caption{Organizzazione dello spazio su un disco in partizioni e filesystem}
513   \label{fig:file_disk_filesys}
514 \end{figure}
515
516 Se si va ad esaminare con maggiore dettaglio la strutturazione
517 dell'informazione all'interno del singolo filesystem (tralasciando i dettagli
518 relativi al funzionamento del filesystem stesso come la strutturazione in
519 gruppi dei blocchi, il superblock e tutti i dati di gestione) possiamo
520 esemplificare la situazione con uno schema come quello esposto in \nfig.
521
522 \begin{figure}[htb]
523   \centering
524   \includegraphics[width=11cm]{img/filesys_struct}
525   \caption{Strutturazione dei dati all'interno di un filesystem}
526   \label{fig:file_filesys_detail}
527 \end{figure}
528
529 Da \curfig\ si evidenziano alcune caratteristiche base di ogni filesystem su
530 cui è bene porre attenzione in quanto sono fondamentali per capire il
531 funzionamento delle funzioni che manipolano i file e le directory su cui
532 torneremo in seguito; in particolare è opportuno ricordare sempre che:
533
534 \begin{enumerate}
535   
536 \item L'\textit{inode} contiene tutte le informazioni riguardanti il file: il
537   tipo di file, i permessi di accesso, le dimensioni, i puntatori ai blocchi
538   fisici che contengono i dati e così via; le informazioni che la funzione
539   \func{stat} fornisce provengono dall'\textit{inode}; dentro una directory si
540   troverà solo il nome del file e il numero dell'\textit{inode} ad esso
541   associato, cioè quella che da qui in poi chiameremo una \textsl{voce}
542   (traduzione approssimata dell'inglese \textit{directory entry}, che non
543   useremo anche per evitare confusione con le \textit{dentry} del kernel di
544   cui si parlava in \secref{sec:file_vfs}).
545   
546 \item Come mostrato in \curfig si possono avere più voci che puntano allo
547   stesso \textit{inode}. Ogni \textit{inode} ha un contatore che contiene il
548   numero di riferimenti (\textit{link count}) che sono stati fatti ad esso;
549   solo quando questo contatore si annulla i dati del file vengono
550   effettivamente rimossi dal disco. Per questo la funzione per cancellare un
551   file si chiama \func{unlink}, ed in realtà non cancella affatto i dati del
552   file, ma si limita a eliminare la relativa voce da una directory e
553   decrementare il numero di riferimenti nell'\textit{inode}.
554   
555 \item Il numero di \textit{inode} nella voce si riferisce ad un \textit{inode}
556   nello stesso filesystem e non ci può essere una directory che contiene
557   riferimenti ad \textit{inodes} relativi ad altri filesystem. Questo limita
558   l'uso del comando \cmd{ln} (che crea una nuova voce per un file
559   esistente, con la funzione \func{link}) al filesystem corrente.
560   
561 \item Quando si cambia nome ad un file senza cambiare filesystem il contenuto
562   del file non deve essere spostato, viene semplicemente creata una nuova voce
563   per l'\textit{inode} in questione e rimossa la vecchia (questa è la modalità
564   in cui opera normalmente il comando \cmd{mv} attraverso la funzione
565   \func{rename}).
566
567 \end{enumerate}
568
569 Infine è bene avere presente che essendo file pure loro, esiste un numero di
570 riferimenti anche per le directory; per cui se ad esempio a partire dalla
571 situazione mostrata in \curfig\ creiamo una nuova directory \file{img} nella
572 directory \file{gapil}: avremo una situazione come quella in \nfig, dove per
573 chiarezza abbiamo aggiunto dei numeri di inode.
574
575 \begin{figure}[htb]
576   \centering 
577   \includegraphics[width=11cm]{img/dir_links}
578   \caption{Organizzazione dei link per le directory}
579   \label{fig:file_dirs_link}
580 \end{figure}
581
582 La nuova directory avrà allora un numero di riferimenti pari a due, in quanto
583 è referenziata dalla directory da cui si era partiti (in cui è inserita la
584 nuova voce che fa riferimento a \file{img}) e dalla voce \file{.}
585 che è sempre inserita in ogni directory; questo vale sempre per ogni directory
586 che non contenga a sua volta altre directory. Al contempo la directory da
587 cui si era partiti avrà un numero di riferiementi di almeno tre, in quanto
588 adesso sarà referenziata anche dalla voce \file{..} di \file{img}.
589
590
591 \subsection{Il filesystem \textsl{ext2}}
592 \label{sec:file_ext2}
593
594 Il filesystem standard usato da Linux è il cosiddetto \textit{second extended
595   filesystem}, identificato dalla sigla \acr{ext2}. Esso supporta tutte le
596 caratteristiche di un filesystem standard unix, è in grado di gestire
597 filename lunghi (256 caratteri, estendibili a 1012), una dimensione fino a
598 4~Tb. 
599
600 Oltre alle caratteristiche standard \acr{ext2} fornisce alcune estensioni
601 che non sono presenti sugli altri filesystem unix, le cui principali sono le
602 seguenti:
603 \begin{itemize}
604 \item i \textit{file attributes} consentono di modificare il comportamento del
605   kernel quando agisce su gruppi di file. Possono essere settati su file e
606   directory e in quest'ultimo caso i nuovi file creati nella directory
607   ereditano i suoi attributi.
608 \item sono supportate entrambe le semantiche di BSD e SVr4 come opzioni di
609   montaggio. La semantica BSD comporta che i file in una directory sono creati
610   con lo stesso identificatore di gruppo della directory che li contiene. La
611   semantica SVr4 comporta che i file vengono creati con l'identificatore del
612   gruppo primario del processo, eccetto il caso in cui la directory ha il bit
613   di \acr{sgid} settato (per una descrizione dettagliata del significato di
614   questi termini si veda \secref{sec:file_access_control}), nel qual caso file
615   e subdirectory ereditano sia il \acr{gid} che lo \acr{sgid}.
616 \item l'amministratore può scegliere la dimensione dei blocchi del filesystem
617   in fase di creazione, a seconda delle sue esigenze (blocchi più grandi
618   permettono un accesso più veloce, ma sprecano più spazio disco).
619 \item il filesystem implementa link simbolici veloci, in cui il nome del file
620   non è salvato su un blocco, ma tenuto all'interno dell'inode (evitando
621   letture multiple e spreco di spazio), non tutti i nomi però possono essere
622   gestiti così per limiti di spazio (il limite è 60 caratteri). 
623 \item vengono supportati i file immutabili (che possono solo essere letti) per
624   la protezione di file di configurazione sensibili, o file
625   \textit{append-only} che possono essere aperti in scrittura solo per
626   aggiungere dati (caratteristica utilizzabile per la protezione dei file di
627   log).
628 \end{itemize}
629
630 La struttura di \acr{ext2} è stata ispirata a quella del filesystem di BSD,
631 un filesystem è composto da un insieme di blocchi, la struttura generale è
632 quella riportata in \figref{fig:file_filesys_detail}, in cui la partizione
633 è divisa in gruppi di blocchi.
634
635 Ciascun gruppo di blocchi contiene una copia delle informazioni essenziali del
636 filesystem (superblock e descrittore del filesystem sono quindi ridondati) per
637 una maggiore affidabilità e possibilità di recupero in caso di corruzione del
638 superblock principale.
639
640 \begin{figure}[htb]
641   \centering
642   \includegraphics[width=9cm]{img/dir_struct}  
643   \caption{Struttura delle directory nel \textit{second extented filesystem}.}
644   \label{fig:file_ext2_dirs}
645 \end{figure}
646
647 L'utilizzo di raggruppamenti di blocchi ha inoltre degli effetti positivi nelle
648 prestazioni dato che viene ridotta la distanza fra i dati e la tabella degli
649 inode. 
650
651 Le directory sono implementate come una linked list con voci di dimensione
652 variabile. Ciascuna voce della lista contiene il numero di inode, la sua
653 lunghezza, il nome del file e la sua lunghezza, secondo lo schema in \curfig;
654 in questo modo è possibile implementare nomi per i file anche molto lunghi
655 (fino a 1024 caratteri) senza sprecare spazio disco.
656
657
658