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