85098c865e948438193276b67debd89c4ae5a85a
[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. Per questo,
75 anche se è usuale parlare di ``file in una directory'' in realtà una directory
76 contiene solo delle etichette per fare riferimento ai file stessi.
77
78 I manuale delle glibc chiama i nomi contenuti nelle directory
79 \textsl{componenti} (in inglese \textit{file name components}), noi li
80 chiameremo più semplicemente nomi. Un file può essere indicato rispetto alla
81 directory corrente semplicemente specificando il nome da essa contenuto. Una
82 directory contiene semplicemente un elenco di questi nomi, che possono
83 corrispondere a un qualunque oggetto del filesystem, compresa un'altra
84 directory; l'albero viene appunto creato inserendo directory in altre
85 directory.
86
87 Il nome completo di file generico è composto da una serie di questi
88 \textsl{componenti} separati da una \texttt{/} (in Linux più \texttt{/}
89 consecutive sono considerate equivalenti ad una sola). Il nome completo di un
90 file viene usualmente chiamato \textit{pathname}, e anche se il manuale della
91 glibc depreca questo nome (poiché genererebbe confusione, dato che con
92 \textit{path} si indica anche un insieme di directory su cui effettuare una
93 ricerca, come quello in cui si cercano i comandi) l'uso è ormai così comune
94 che è senz'altro più chiaro dell'alternativa proposta.
95
96 Il processo con cui si associa ad un pathname uno specifico file è chiamato
97 risoluzione del nome (\textit{file name resolution} o \textit{pathname
98   resolution}).  La risoluzione viene fatta esaminando il pathname da destra a
99 sinistra e localizzando ogni nome nella directory indicata dal nome
100 precedente: ovviamente perché il procedimento funzioni occorre che i nomi
101 indicati come directory esistano e siano effettivamente directory, inoltre i
102 permessi devono consentire l'accesso.
103
104 Se il pathname comincia per \texttt{/} la ricerca parte dalla directory radice
105 del processo; questa, a meno di un \textit{chroot} (su cui torneremo in
106 seguito, vedi \secref{sec:xxx_chroot}) è la stessa per tutti i processi ed
107 equivale alla directory radice dell'albero (come descritto in
108 \secref{sec:fileintr_organization}): in questo caso si parla di un pathname
109 \textsl{assoluto}. Altrimenti la ricerca parte dalla directory corrente (su
110 cui torneremo più avanti in \secref{sec:filedir_work_dir}) ed il pathname è
111 detto \textsl{relativo}.
112
113 I nomi \texttt{.} e \texttt{..} hanno un significato speciale e vengono
114 inseriti in ogni directory, il primo fa riferimento alla directory corrente e
115 il secondo alla directory \textsl{genitore} (\textit{parent directory}) cioè
116 la directory che contiene il riferimento alla directory corrente; nel caso
117 questa sia la directory radice allora il riferimento è a se stessa.
118
119
120 % \subsection{Il controllo di accesso}
121 % \label{sec:fileintr_access_ctrl}
122
123
124 \subsection{I tipi di files}
125 \label{sec:fileintr_file_types}
126
127 Come detto in precedenza in unix esistono vari tipi di file, in Linux questi
128 sono implementati come oggetti del \textit{Virtual File System} (vedi
129 \secref{sec:fileintro_vfs}) e sono presenti in tutti i filesystem unix-like
130 utilizzabili con Linux. L'elenco dei vari tipi di file definiti dal Virtual
131 File System è riportato in \ntab.
132
133 Si tenga ben presente che questa classificazione non ha nulla a che fare con
134 la classificazione sui tipi di file (che in questo caso sono sempre file di
135 dati) in base al loro contenuto, o tipo di accesso.
136
137 \begin{table}[htb]
138   \begin{center}
139     \begin{tabular}[c]{l l p{7cm}}
140     \multicolumn{2}{c}{\textbf{Tipo di file}} & \textbf{Descrizione} \\
141     \hline
142       \textit{regular file} & \textsl{file normale} &
143       un file che contiene dei dati (l'accezione normale di file) \\
144       \textit{directory} & \textsl{cartella o direttorio} &
145       un file che contiene una lista di nomi associati a degli inodes \\
146       \textit{symbolic link} & \textsl{collegamento simbolico} &
147       un file che contiene un riferimento ad un altro file/directory \\
148       \textit{char device} & \textsl{dispositivo a caratteri} &
149       un file che identifica una periferica ad accesso sequenziale \\
150       \textit{block device} & \textsl{dispositivo a blocchi} &
151       un file che identifica una periferica ad accesso diretto \\
152       \textit{fifo} & \textsl{tubo} &
153       un file speciale che identifica una linea di comunicazione software
154       (unidirezionale) \\
155       \textit{socket} & \textsl{manicotto} &
156       un file speciale che identifica una linea di comunicazione software
157       (bidirezionale) \\
158     \hline
159     \end{tabular}
160     \caption{Tipologia dei file definiti nel VFS}
161     \label{tab:fileintr_file_types}
162   \end{center}
163 \end{table}
164
165 Una delle differenze principali con altri sistemi operativi (come il VMS o
166 Windows) è che per Unix tutti i file di dati sono identici e contengono un
167 flusso continuo di bytes; non esiste cioè differenza per come vengono visti
168 dal sistema file di diverso contenuto o formato (come nel caso di quella fra
169 file di testo e binari che c'è in Windows) né c'è una strutturazione a record
170 per il cosiddetto ``accesso diretto'' come nel caso del VMS.
171 %  (con i kernel
172 % della serie 2.4 è disponibile una forma di accesso diretto ai dischi il
173 % \textit{raw access} che però non ha nulla a che fare con questo).
174
175 Una seconda differenza è nel formato dei file ASCII; in Unix la fine riga è
176 codificata in maniera diversa da Windows o MacIntosh, in particolare il fine
177 riga è il carattere \texttt{LF} (o \verb|\n|) al posto del \texttt{CR}
178 (\verb|\r|) del mac e del \texttt{CR LF} di Windows. Questo può causare alcuni
179 problemi qualora nei programmi si facciano assunzioni sul terminatore della
180 riga.
181
182
183 \subsection{Le due interfacce ai file}
184 \label{sec:fileintr_io_api}
185
186 In unix le modalità di accesso ai file e le relative interfacce di
187 programmazione sono due, basate su due diversi meccanismi con cui è possibile
188 accedere al loro contenuto.
189
190 La prima è l'interfaccia standard di unix, quella che il manuale delle glibc
191 chiama interfaccia dei descrittore di file (o \textit{file descriptor}).  È
192 un'interfaccia specifica di unix e provvede un accesso non bufferizzato.
193
194 L'interfaccia è primitiva ed essenziale, l'accesso viene detto non
195 bufferizzato in quanto la lettura e la scrittura vengono eseguite chiamando
196 direttamente le system call del kernel (in realtà il kernel effettua al suo
197 interno alcune bufferizzazioni per aumentare l'efficienza nell'accesso ai
198 dispositivi); i file descriptors sono rappresentati da numeri interi (cioè
199 semplici variabili di tipo \texttt{int}).  L'interfaccia è definita
200 nell'header \texttt{unistd.h}.
201
202 La seconda interfaccia è quella che il manuale della glibc chiama degli
203 \textit{stream}, essa provvede funzioni più evolute e un accesso bufferizzato
204 (controllato dalla implementazione fatta dalle librerie del C).  Questa è
205 l'interfaccia standard usata dal linguaggio C e perciò si trova anche su tutti
206 i sistemi non Unix. Gli stream sono oggetti complessi e sono rappresentati da
207 puntatori ad un opportuna struttura definita dalle librerie del C, si accede
208 ad essi sempre in maniera indiretta utilizzando il tipo \texttt{FILE *}.
209 L'interfaccia è definita nell'header \texttt{stdio.h}.
210
211 Entrambe le interfacce possono essere usate per l'accesso ai file come agli
212 altri oggetti del VFS (pipes, socket, device), ma per poter accedere alle
213 operazioni di controllo sul particolare tipo di oggetto del VFS scelto occorre
214 usare l'interfaccia standard di unix coi file descriptors. Allo stesso modo
215 devono essere usati i file descriptor se si vuole ricorrere a modalità
216 speciali di I/O come il polling o il non-bloccante (vedi
217 \secref{sec:file_xxx}).
218
219 Gli stream forniscono un'interfaccia di alto livello costruita sopra quella
220 dei file descriptor, che tratta tutti i file nello stesso modo, con
221 l'eccezione di poter scegliere tra diversi stili di bufferizzazione.  Il
222 maggior vantaggio degli stream è che l'interfaccia per le operazioni di
223 input/output è enormemente più ricca di quella dei file descriptor, che
224 provvedono solo funzioni elementari per la lettura/scrittura diretta di
225 blocchi di bytes.  In particolare gli stream dispongono di tutte le funzioni
226 di formattazione per l'input e l'output adatte per manipolare anche i dati in
227 forma di linee o singoli caratteri.
228
229 In ogni caso, dato che gli stream sono implementati sopra l'interfaccia
230 standard di unix, è sempre possibile estrarre il file descriptor da uno stream
231 ed eseguirvi operazioni di basso livello, o associare in un secondo tempo uno
232 stream ad un file descriptor.
233
234 In generale, se non necessitano specificatamente le funzionalità di basso
235 livello, è opportuno usare sempre gli stream per la loro maggiore portabilità
236 essendo questi ultimi definiti nello standard ANSI C; l'interfaccia con i file
237 descriptor invece segue solo lo standard POSIX.1 dei sistemi unix ed è
238 pertanto di portabilità più limitata.
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 ridirigendo 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, queste funzioni 
349
350
351
352 \subsection{Il funzionamento del VFS}
353 \label{sec:fileintr_vfs_work}
354
355 La funzione più importante implementata dal VFS è la system call \texttt{open}
356 che permette di aprire un file. Dato un pathname viene eseguita una ricerca
357 dentro la \textit{directory entry cache} (in breve \textit{dcache}),
358 una tabella di hash che contiene tutte le \textit{directory entry} (in breve
359 \textit{dentry}) che permette di associare in maniera rapida ed efficiente il
360 pathname a una specifica dentry.
361
362 Una singola dentry contiene in genere il puntatore ad un \textit{inode};
363 quest'ultimo è la struttura base che sta sul disco e che identifica un singolo
364 oggetto del VFS sia esso un file ordinario, una directory, una FIFO, un file
365 di dispositivo, o una qualsiasi altra cosa che possa essere rappresentata dal
366 VFS (sui tipi di ``files'' possibili torneremo in seguito). A ciascuno di essi
367 è associata pure una struttura che sta in memoria, e che oltre alle
368 informazioni sullo specifico file contiene pure il riferimento alle funzioni
369 (i \textsl{metodi}) da usare per poterlo manipolare.
370
371 Le dentries ``vivono'' in memoria e non vengono mai salvate su disco, vengono
372 usate per motivi di velocità, gli inodes invece stanno su disco e vengono
373 copiati in memoria quando serve, ed ogni cambiamento viene copiato
374 all'indietro sul disco, gli inodes che stanno in memoria sono inodes del VFS
375 ed è ad essi che puntano le singole dentry.
376
377 La dcache costituisce perciò una sorta di vista completa di tutto l'albero dei
378 files, ovviamente per non riempire tutta la memoria questa vista è parziale
379 (la dcache cioè contiene solo le dentry per i file per i quali è stato
380 richiesto l'accesso), quando si vuole risolvere un nuovo pathname il VFS deve
381 creare una nuova dentry e caricare l'inode corrispondente in memoria. 
382
383 Questo procedimento viene eseguito dal metodo \texttt{lookup()} dell'inode
384 della directory che contiene il file; questo viene installato nelle relative
385 strutture in memoria quando si effettua il montaggio lo specifico filesystem
386 su cui l'inode va a vivere.
387
388 Una volta che il VFS ha a disposizione la dentry (ed il relativo inode)
389 diventa possibile accedere alle varie operazioni sul file come la
390 \texttt{open} per aprire il file o la \texttt{stat} per leggere i dati
391 dell'inode e passarli in user space.
392
393 L'apertura di un file richiede comunque un'altra operazione, l'allocazione di
394 una struttura di tipo \texttt{file} in cui viene inserito un puntatore alla
395 dentry e una struttura \verb|f_ops| che contiene i puntatori ai metodi che
396 implementano le operazioni disponibili sul file. In questo modo i processi in
397 user space possono accedere alle operazioni attraverso detti metodi, che
398 saranno diversi a seconda del tipo di file (o dispositivo) aperto (su questo
399 torneremo in dettaglio in \secref{sec:fileunix_fd}). Un elenco delle operazioni
400 previste dal kernel è riportato in \ntab.
401
402 \begin{table}[htb]
403   \centering
404   \begin{tabular}[c]{c p{7cm}}
405     \textbf{funzione} & \textbf{operazione} \\
406     \hline
407     \textit{open}    & apre il file \\
408     \textit{read}    & legge dal file \\
409     \textit{write}   & scrive sul file \\ 
410     \textit{llseek}  & sposta la posizione corrente sul file \\
411     \textit{ioctl}   & accede alle operazioni di controllo 
412                        (tramite la \texttt{ioctl})\\
413     \textit{readdir} & per leggere il contenuto di una directory \\
414     \textit{poll}    & \\
415     \textit{mmap}    & chiamata dalla system call \texttt{mmap}. 
416                        mappa il file in memoria\\
417     \textit{release} & chiamata quando l'ultima referenza a un file 
418                        aperto è chiusa\\
419     \textit{fsync}   & chiamata dalla system call \texttt{fsync} \\
420     \textit{fasync}  & chiamate da \texttt{fcntl} quando è abilitato 
421                        il modo asincrono per l'I/O su file. \\
422     \hline
423   \end{tabular}
424   \caption{Operazioni sui file definite nel VFS.}
425   \label{tab:fileintr_file_operations}
426 \end{table}
427
428 In questo modo per ciascun file diventano utilizzabili una serie di operazioni
429 (non è dette che tutte siano disponibili), che costituiscono l'interfaccia
430 astratta del VFS, e qualora se ne voglia eseguire una il kernel andrà ad
431 utilizzare la opportuna routine dichiarata in \verb|f_ops| appropriata al tipo
432 di file in questione. 
433
434 Così sarà possibile scrivere sulla porta seriale come su un file di dati
435 normale; ovviamente certe operazioni (nel caso della seriale ad esempio la
436 \textit{seek}) non saranno disponibili, però con questo sistema l'utilizzo di
437 diversi filesystem (come quelli usati da Windows o MacOs) è immediato e
438 (relativamente) trasparente per l'utente ed il programmatore.
439
440
441 \subsection{Il funzionamento di un filesystem unix}
442 \label{sec:fileintr_filesystem}
443
444 Come già accennato in \secref{sec:fileintr_organization} Linux (ed ogni unix
445 in generale) organizza i dati che tiene su disco attraverso l'uso di un
446 filesystem. Una delle caratteristiche di Linux rispetto agli altri unix è
447 quella di poter supportare grazie al VFS una enorme quantità di filesystem
448 diversi, ognuno dei quali ha una sua particolare struttura e funzionalità
449 proprie; per questo non entreremo nei dettagli di un filesystem specifico, ma
450 daremo una descrizione a grandi linee che si adatta alle caratteristiche
451 comuni di un qualunque filesystem standard unix.
452
453 Dato un disco lo spazio fisico viene usualmente diviso in partizioni; ogni
454 partizione può contenere un filesystem; quest'ultimo è in genere strutturato
455 secondo \nfig, con una lista di inodes all'inizio e il resto dello spazio a
456 disposizione per i dati e le directory.
457
458 \begin{figure}[htb]
459   \centering
460   
461   \caption{Organizzazione dello spazio su un disco in partizioni e filesystem}
462   \label{fig:fileintr_disk_filesys}
463 \end{figure}
464
465 Se si va ad esaminare come è strutturata l'informazione all'interno di un
466 singolo filesystem (tralasciando le parti connesse alla strutturazione e al
467 funzionamento del filesystem stesso come il super-block) avremo una situazione
468 del tipo di quella esposta in \nfig.
469 \begin{figure}[htb]
470   \centering
471   
472   \caption{Organizzazione di un filesystem}
473   \label{fig:fileintr_filesys_detail}
474 \end{figure}
475 da questa figura si evidenziano alcune caratteristiche su cui è bene porre
476 attenzione in quanto sono fondamentali per capire il funzionamento delle
477 funzioni che manipolano i file e le directory su cui torneremo fra poco; in
478 particolare è opportuno ricordare sempre che:
479
480 \begin{enumerate}
481   
482 \item L'\textit{inode} contiene tutte le informazioni riguardanti il file: il
483   tipo di file, i permessi di accesso, le dimensioni, i puntatori ai blocchi
484   fisici che contengono i dati e così via; le informazioni che la funzione
485   \texttt{stat} fornisce provengono dall'\textit{inode}; dentro una directory
486   si troverà solo il nome del file e il numero dell'\textit{inode} ad esso
487   associato, cioè quella che da qui in poi chiameremo una \textsl{voce}
488   (traduzione approssimata dell'inglese \textit{directory entry}, che non
489   useremo anche per evitare confusione con le \textit{dentries} del kernel di
490   cui si parlava in \secref{sec:fileintr_vfs}).
491   
492 \item Come mostrato in \curfig si possono avere più voci che puntano allo
493   stesso \textit{inode}. Ogni \textit{inode} ha un contatore che contiene il
494   numero di riferimenti (\textit{link count}) che sono stati fatti ad esso;
495   solo quando questo contatore si annulla i dati del file vengono
496   effettivamente rimossi dal disco. Per questo la funzione per cancellare un
497   file si chiama \texttt{unlink}, ed in realtà non cancella affatto i dati del
498   file, ma si limita a eliminare la relativa voce da una directory e
499   decrementare il numero di riferimenti nell'\textit{inode}.
500   
501 \item Il numero di \textit{inode} nella voce si riferisce ad un \textit{inode}
502   nello stesso filesystem e non ci può essere una directory che contiene
503   riferimenti ad \textit{inodes} relativi ad altri filesystem. Questo limita
504   l'uso del comando \texttt{ln} (che crea una nuova voce per un file
505   esistente, con la funzione \texttt{link}) al filesystem corrente.
506   
507 \item Quando si cambia nome ad un file senza cambiare filesystem il contenuto
508   del file non deve essere spostato, viene semplicemente creata una nuova voce
509   per l'\textit{inode} in questione e rimossa la vecchia (questa è la modalità
510   in cui opera normalmente il comando \texttt{mv} attraverso la funzione
511   \texttt{rename}).
512
513 \end{enumerate}
514
515 Infine è bene avere presente che essendo file pure loro, esiste un numero di
516 riferimenti anche per le directories; per cui se ad esempio a partire dalla
517 situazione mostrata in \curfig\ creiamo una nuova directory \texttt{textdir}
518 nella directory corrente avremo una situazione come quella in \nfig, dove per
519 chiarezza abbiamo aggiunto dei numeri di inode.
520
521 La nuova directory avrà allora un numero di riferimenti pari a due, in quanto
522 è referenziata dalla directory da cui si era partiti (in cui è inserita la
523 nuova voce che fa riferimento a \texttt{textdir}) e dalla voce \texttt{.}
524 che è sempre inserita in ogni directory; questo vale sempre per ogni directory
525 che non contenga a sua volta altre directories. Al contempo la directory da
526 cui si era partiti avrà un numero di riferiementi di almeno tre, in quanto
527 adesso sarà referenziata anche dalla voce \texttt{..} di \texttt{textdir}.
528
529
530 \subsection{Le funzioni \texttt{link} e \texttt{unlink}}
531 \label{sec:fileintr_link}
532
533 Una delle caratteristiche usate quando si opera con i file è quella di poter
534 creare dei nomi fittizi (alias o collegamenti) per potersi riferire allo
535 stesso file accedendovi da directory diverse. Questo è possibile anche in
536 ambiente unix, dove tali collegamenti sono usualmente chiamati \textit{link},
537 ma data la struttura del sistema ci sono due metodi sostanzialmente diversi
538 per fare questa operazione.
539
540 Come si è appena detto l'accesso al contenuto di un file su disco avviene
541 attraverso il suo inode, e il nome che si trova in una directory è solo una
542 etichetta associata ad un puntatore a detto inode.  Questo significa che la
543 realizzazione di un link è immediata in quanto uno stesso file può avere tanti
544 nomi diversi allo stesso tempo, dati da altrettante diverse associazioni allo
545 stesso inode; si noti poi che nessuno di questi nomi viene ad assumere una
546 particolare preferenza rispetto agli altri.
547
548 Per aggiungere un nome ad un inode si utilizza la funzione \texttt{link}; si
549 suole chiamare questo tipo di associazione un collegamento diretto (o
550 \textit{hard link}).  Il prototipo della funzione e le sue caratteristiche
551 principali, come risultano dalla man page, sono le seguenti:
552 \begin{prototype}{unistd.h}
553 {int link(const char * oldpath, const char * newpath)}
554   Crea un nuovo collegamento diretto al file indicato da \texttt{oldpath}
555   dandogli nome \texttt{newpath}.
556   
557   La funzione restituisce zero in caso di successo e -1 per un errore, in caso
558   di errore. La variabile \texttt{errno} viene settata secondo i seguenti
559   codici di errore:
560   \begin{errlist}
561   \item \texttt{EXDEV} \texttt{oldpath} e \texttt{newpath} non sono sullo
562     stesso filesystem.
563   \item \texttt{EPERM} il filesystem che contiene \texttt{oldpath} e
564     \texttt{newpath} non supporta i link diretti o è una directory.
565   \item \texttt{EFAULT} una delle stringhe passate come parametri è fuori
566     dello spazio di indirizzi del processo.
567   \item \texttt{EACCESS} errore di accesso (mancano i permessi per scrivere o
568     per attraversare le directories), vedi \secref{sec:filedir_access_control}
569     per i dettagli.
570   \item \texttt{ENAMETOOLONG} una dei due pathname è troppo lungo.
571   \item \texttt{ENOENT} un componente di \texttt{oldpath} o \texttt{newpath}
572     non esiste o è un link simbolico spezzato.
573   \item \texttt{ENOTDIR} un componente di \texttt{oldpath} o \texttt{newpath}
574     non è una directory.
575   \item \texttt{ENOMEM} il kernel non ha a disposizione memoria sufficiente a
576     completare l'operazione. 
577   \item \texttt{EROFS} la directory su cui si vuole inserire il nuovo link è
578     su un filesystem montato readonly.
579   \item \texttt{EEXIST} un file (o una directory) con quel nome esiste di
580     già.
581   \item \texttt{EMLINK} ci sono troppi link al file \texttt{oldpath} (il
582     numero massimo è specificato dalla variabile \texttt{LINK\_MAX}, vedi
583     \secref{sec:xxx_limits}).
584   \item \texttt{ELOOP} si incontrati troppi link simbolici nella risoluzione
585     di \texttt{oldpath} o \texttt{newpath}.
586   \item \texttt{ENOSPC} la directory in cui si vuole creare il link non ha
587     spazio per ulteriori voci.
588   \item \texttt{EIO} c'è stato un errore di input/output.
589   \end{errlist}
590 \end{prototype}
591
592 La creazione di un nuovo collegamento diretto non copia il contenuto del file,
593 ma si limita ad aumentare di uno il numero di referenze al file aggiungendo il
594 nuovo nome ai precedenti. Si noti che uno stesso file può essere così
595 richiamato in diverse directory.
596  
597 Per quanto dicevamo in \secref{sec:fileintr_filesystem} la creazione del
598 collegamento diretto è possibile solo se entrambi i pathname sono nello stesso
599 filesystem; inoltre il filesystem deve supportare i collegamenti diretti (non è
600 il caso ad esempio del filesystem \texttt{vfat} di windows).
601
602 La funzione opera sui file ordinari, come sugli altri oggetti del filesystem,
603 ma solo l'amministratore è in grado di creare un collegamento diretto ad
604 un'altra directory, questo lo si fa perché in questo caso è possibile creare
605 dei circoli nel filesystem (vedi \secref{sec:fileintr_symlink}) che molti
606 programmi non sono in grado di gestire e la cui rimozione diventa estremamente
607 complicata (in genere occorre far girare il programma \texttt{fsck} per
608 riparare il filesystem); data la sua pericolosità in Linux questa
609 caratteristica è stata disabilitata, e la funzione restituisce l'errore
610 \texttt{EPERM}.
611
612 La rimozione di un file (o più precisamente della voce che lo referenzia) si
613 effettua con la funzione \texttt{unlink}; il suo prototipo è il seguente:
614
615 \begin{prototype}{unistd.h}{int unlink(const char * pathname)}
616   Cancella il nome specificato dal pathname nella relativa directory e
617   decrementa il numero di riferimenti nel relativo inode. Nel caso di link
618   simbolico cancella il link simbolico; nel caso di socket, fifo o file di
619   dispositivo rimuove il nome, ma come per i file i processi che hanno aperto
620   uno di questi oggetti possono continuare ad utilizzarlo.
621   
622   La funzione restituisce zero in caso di successo e -1 per un errore, nel
623   qual caso il file non viene toccato. La variabile \texttt{errno} viene
624   settata secondo i seguenti codici di errore:
625   \begin{errlist}
626   \item \texttt{EACCESS} errore di accesso (mancano i permessi per scrivere o
627     per attraversare le directories), vedi \secref{sec:filedir_access_control}
628     per i dettagli.
629   \item \texttt{EISDIR} \texttt{pathname} si riferisce ad una directory
630     (valore specifico ritornato da linux che non consente l'uso di
631     \texttt{unlink} con le directory, e non conforme allo standard POSIX, che
632     prescrive invece l'uso di \texttt{EPERM} in caso l'operazione non sia
633     consnetita o il processo non abbia privilegi sufficienti).
634   \item \texttt{EFAULT} la stringa
635     passata come parametro è fuori dello spazio di indirizzi del processo.
636   \item \texttt{ENAMETOOLONG} il pathname troppo lungo.
637   \item \texttt{ENOENT} uno dei componenti del pathname non esiste o è un link
638     simbolico spezzato.
639   \item \texttt{ENOTDIR} uno dei componenti del pathname non è una directory.
640   \item \texttt{EISDIR} \texttt{pathname} fa riferimento a una directory.
641   \item \texttt{ENOMEM} il kernel non ha a disposizione memoria sufficiente a
642     completare l'operazione. 
643   \item \texttt{EROFS} \texttt{pathname} è su un filesystem montato in sola
644     lettura.
645   \item \texttt{ELOOP} ci sono troppi link simbolici nella risoluzione del
646     pathname.
647   \item \texttt{EIO} errore di input/output.
648   \end{errlist}
649 \end{prototype}
650
651 Per cancellare una voce in una directory è necessario avere il permesso di
652 scrittura su di essa (dato che si va a rimuovere una voce dal suo contenuto) e
653 il diritto di esecuzione sulla directory che la contiene (torneremo in
654 dettaglio sui permessi e gli attributi fra poco), se inoltre lo
655 \textit{sticky} bit è settato occorrerà anche essere proprietari del file o
656 proprietari della directory (o root, per cui nessuna delle restrizioni è
657 applicata).
658
659 Una delle caratteristiche di queste funzioni è che la creazione/rimozione
660 della nome dalla directory e l'incremento/decremento del numero di riferimenti
661 nell'inode deve essere una operazione atomica (cioè non interrompibile da
662 altri) processi, per questo entrambe queste funzioni sono realizzate tramite
663 una singola system call.
664
665 Si ricordi infine che il file non viene eliminato dal disco fintanto che tutti
666 i riferimenti ad esso sono stati cancellati, solo quando il \textit{link
667   count} mantenuto nell'inode diventa zero lo spazio occupato viene rimosso. A
668 questo però si aggiunge una altra condizione, e cioè che non ci siano processi
669 che abbiano detto file aperto. Come accennato questa proprietà viene spesso
670 usata per essere sicuri di non lasciare file temporanei su disco in caso di
671 crash dei programmi; la tecnica è quella di aprire il file e chiamare
672 \texttt{unlink} subito dopo.
673
674 \subsection{Le funzioni \texttt{remove} e \texttt{rename}}
675 \label{sec:fileintr_remove}
676
677 Al contrario di quanto avviene con altri unix in Linux non è possibile usare
678 \texttt{unlink} sulle directory, per cancellare una directory si può usare la
679 funzione \texttt{rmdir} (vedi \secref{sec:filedir_dir_creat_rem}), oppure la
680 funzione \texttt{remove}. Questa è la funzione prevista dallo standard ANSI C
681 per cancellare un file o una directory (e funziona anche per i sistemi che non
682 supportano i link diretti), che per i file è identica alla \texttt{unlink} e
683 per le directory è identica alla \texttt{rmdir}:
684
685 \begin{prototype}{stdio.h}{int remove(const char *pathname)}
686   Cancella un nome dal filesystem. Usa \texttt{unlink} per i file e
687   \texttt{rmdir} per le directory.
688   
689   La funzione restituisce zero in caso di successo e -1 per un errore, nel
690   qual caso il file non viene toccato. Per i codici di errori vedi quanto
691   riportato nella descrizione di \texttt{unlink} e \texttt{rmdir}.
692 \end{prototype}
693
694 Per cambiare nome ad un file si usa invece la funzione \texttt{rename}, il
695 vantaggio nell'uso di questa funzione al posto della chiamata successiva di
696 \texttt{unlink} e \texttt{link} è che l'operazione è eseguita atomicamente, in
697 questo modo non c'è la possibilità che un processo che cerchi di accedere al
698 nuovo nome dopo che il vecchio è stato cambiato lo trovi mancante.
699
700 \begin{prototype}{stdio.h}
701 {int rename(const char *oldpath, const char *newpath)}
702   Rinomina un file, spostandolo fra directory diverse quando richiesto.
703
704   La funzione restituisce zero in caso di successo e -1 per un errore, nel
705   qual caso il file non viene toccato. La variabile \texttt{errno} viene
706   settata secondo i seguenti codici di errore:
707   \begin{errlist} 
708   \item \texttt{EISDIR} \texttt{newpath} è una directory già esistente mentre
709     \texttt{oldpath} non è una directory. 
710   \item \texttt{EXDEV} \texttt{oldpath} e \texttt{newpath} non sono sullo
711     stesso filesystem. 
712   \item \texttt{ENOTEMPTY} \texttt{newpath} è una directory già esistente e
713     non vuota.
714   \item \texttt{EBUSY} o \texttt{oldpath} o \texttt{newpath} sono in uso da
715     parte di qualche processo (come directory di lavoro o come root) o del
716     sistema (come mount point).
717   \item \texttt{EINVAL} \texttt{newpath} contiene un prefisso di
718     \texttt{oldpath} o più in generale si è cercato di creare una directory
719     come sottodirectory di se stessa.
720   \item \texttt{EMLINK} \texttt{oldpath} ha già il massimo numero di link
721     consentiti o è una directory e la directory che contiene \texttt{newpath}
722     ha già il massimo numero di link. 
723   \item \texttt{ENOTDIR} Uno dei componenti dei pathname non è una directory
724     o\texttt{oldpath} è una directory e \texttt{newpath} esiste e non è una
725     directory.
726   \item \texttt{EFAULT} o \texttt{oldpath} o \texttt{newpath} è fuori dello
727     spazio di indirizzi del processo.
728   \item \texttt{EACCESS} Non c'è il permesso di scrittura per la directory in
729     cui si vuole creare il nuovo link o una delle directory del pathname non
730     consente la ricerca (permesso di esecuzione).
731   \item \texttt{EPERM} le directory contenenti \texttt{oldpath} o
732     \texttt{newpath} hanno lo sticky bit attivo e i permessi del processo non
733     consentono rispettivamente la cancellazione e la creazione del file, o il
734     filesystem non supporta i link.
735   \item \texttt{ENAMETOOLONG} uno dei pathname è troppo lungo.
736   \item \texttt{ENOENT} Uno dei componenti del pathname non esiste o è un link
737     simbolico spezzato.
738   \item \texttt{ENOMEM} il kernel non ha a disposizione memoria sufficiente a
739     completare l'operazione. 
740   \item \texttt{EROFS} I file sono su un filesystem montato in sola lettura.
741   \item \texttt{ELOOP} Ci sono troppi link simbolici nella risoluzione del
742     pathname.
743   \item \texttt{ENOSPC} Il device di destinazione non ha più spazio per la
744     nuova voce. 
745   \end{errlist}    
746 \end{prototype}
747
748 \subsection{I link simbolici}
749 \label{sec:fileintr_symlink}
750
751 Siccome la funzione \texttt{link} crea riferimenti agli inodes, essa può
752 funzionare soltanto per file che risiedono sullo stesso filesystem, dato che
753 in questo caso è garantita l'unicità dell'inode, e solo per un filesystem di
754 tipo unix.  Inoltre in Linux non è consentito eseguire un link diretto ad una
755 directory.
756
757 Per ovviare a queste limitazioni i sistemi unix supportano un'altra forma di
758 link (i cosiddetti \textit{soft link} o \textit{symbolic link}), che sono,
759 come avviene in altri sistemi operativi, dei file che contengono il
760 semplicemente il riferimento ad un altro file (o directory). In questo modo è
761 possibile effettuare link anche attraverso filesystem diversi e a directory, e
762 pure a file che non esistono ancora.
763
764 Il sistema funziona in quanto i link simbolici sono contrassegnati come tali
765 al kernel (analogamente a quanto avviene per le directory) per cui la chiamata
766 ad una \texttt{open} o una \texttt{stat} su un link simbolico comporta la
767 lettura del contenuto del medesimo e l'applicazione della funzione al file
768 specificato da quest'ultimo. Invece altre funzioni come quelle per cancellare
769 o rinominare i file operano direttamente sul link simbolico. Inoltre esistono
770 funzioni apposite, come la \texttt{readlink} e la \texttt{lstat} per accedere
771 alle informazioni del link invece che a quelle del file a cui esso fa
772 riferimento.
773
774 Le funzioni per operare sui link simbolici sono le seguenti, esse sono tutte
775 dichiarate nell'header file \texttt{unistd.h}.
776
777 \begin{prototype}{unistd.h}
778 {int symlink(const char * oldname, const char * newname)}
779   Crea un nuovo link simbolico al file indicato da \texttt{oldname} dandogli
780   nome \texttt{newname}.
781   
782   La funzione restituisce zero in caso di successo e -1 per un errore, in caso
783   di errore. La variabile \texttt{errno} viene settata secondo i codici di
784   errore standard di accesso ai files (trattati in dettaglio in
785   \secref{sec:filedir_access_control}) ai quali si aggiungono i seguenti:
786   \begin{errlist}
787   \item \texttt{EEXIST} Un file (o una directory) con quel nome esiste di
788     già.
789   \item \texttt{EROFS} La directory su cui si vuole inserire il nuovo link è
790     su un filesystem montato readonly.
791   \item \texttt{ENOSPC} La directory o il filesystem in cui si vuole creare il
792     link è piena e non c'è ulteriore spazio disponibile.
793   \item \texttt{ELOOP} Ci sono troppi link simbolici nella risoluzione di
794     \texttt{oldname} o di \texttt{newname}.
795   \end{errlist}
796 \end{prototype}
797
798 Dato che la funzione \texttt{open} segue i link simbolici, è necessaria usare
799 un'altra funzione quando si vuole leggere il contenuto di un link simbolico,
800 questa funzione è la:
801
802 \begin{prototype}{unistd.h}
803 {int readlink(const char * path, char * buff, size\_t size)} 
804   Legge il contenuto del link simbolico indicato da \texttt{path} nel buffer
805   \texttt{buff} di dimensione \texttt{size}. Non chiude la stringa con un
806   carattere nullo e la tronca a \texttt{size} nel caso il buffer sia troppo
807   piccolo per contenerla.
808   
809   La funzione restituisce il numero di caratteri letti dentro \texttt{buff} o
810   -1 per un errore, in caso di errore. La variabile \texttt{errno} viene
811   settata secondo i codici di errore:
812   \begin{errlist}
813   \item \texttt{EEXIST} Un file (o una directory) con quel nome esiste di
814     già.
815   \item \texttt{EROFS} La directory su cui si vuole inserire il nuovo link è
816     su un filesystem montato readonly.
817   \item \texttt{ENOSPC} La directory o il filesystem in cui si vuole creare il
818     link è piena e non c'è ulteriore spazio disponibile.
819   \item \texttt{ELOOP} Ci sono troppi link simbolici nella risoluzione di
820     \texttt{oldname} o di \texttt{newname}.
821   \end{errlist}
822 \end{prototype}
823
824