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