System plików Unix
Transkrypt
System plików Unix
Algorytm przydzielania bloku getblk( ) input: output: file system number block number locked buffer that can now be used for block { while (buffer not found) { if (block in hash queue) { if (buffer busy) /* scenario 5 */ { sleep (event buffer becomes free); continue; /* back to while loop */ } lock buffer; /* scenario 1 */ remove buffer from free list; return buffer; } else /* block not on hash queue */ { if (no free buffers) /* scenario 4 */ { sleep (event any buffer becomes free); continue; /* back to white loop */ } remove buffer from free list; if (buffer marked for delayed write) { /* scenario 3 */ asynchronous write buffer to disk; continue; /* back to while loop */ } /* scenario 2--found a free buffer */ remove buffer from old hash queue; put buffer onto new hash queue; return buffer; } } } Algorytm zwalniania bufora brelse( ) input: locked buffer output: none { wakeup( all procs, waiting for any buffer to become free); wakeup( all procs, waiting for this buffer to become free); raise processor execution level to block interrupts; if (buffer contents valid && buffer not old) enqueue buffer at end of free list; else enqueue buffer at beginning of free list; lower processor execution level to allow interrupts; unlock(buffer); } Algorytmy czytania bloku 1. czytanie pojedynczego bloku bread( ) input: block number output: buffer containing data { getblk(block number); if (buffer data valid) return buffer; initiate disk read; sleep(event disk read complete); return (buffer); } /*synchronous*/ 2. czytanie bloku z “look-ahead” breada() input: output: block number for immediate read block number for asynchronous read buffer containing data for immeciiate read { if (first block not in cache) { getblk(first block); if (buffer data not valid) initiate disk read; } if (second block not in cache) { getbl k(second block); if (buffer data valid) brelse(second block); else initiate disk read; } if (first block was originally in cache) { bread (first block); return buffer; } sleep (event first buffer contains valid data); return buffer; } /*synchronous*/ /*asynchronous*/ Algorytm zapisu bloku bwrite( ) input: buffer output: none { initiate disk write; if (l/O synchronous) { sleep(event I/O complete); brelse(buffer); } else if (buffer marked for delayed write mark buffer to put at head of free list; } Algorytm czytania INODE iget() input: file system inode number output: locked inode { while (not done) { if (inode in inode cache) { if (inode locked) { sleep (event inode becomes unlocked); continue; /* loop */ } /*tutaj ominiety kod dot. wej if (inode on inode free list) remove from free list; increment inode reference count; return (inode); } /* inode not in inode cache */ if (no inodes on free list) return(error); remove new inode from free list, reset inode number and file system; remove inode from old hash queue, place on new one; bread(inode block); initialize inode (reference count = 1); return (inode); } } Zwalnianie kopii INODE w pamieci iput() input: pointer to in-core inode output: none { lock inode if not already locked; decrement inode reference count; if (reference count==0) { if(inode link count==0) { free disk blocks for file (algorithm free); set file type to 0; ifree(inode); } if (file accessed or inode changed or file changed) update disk inode; put inode on free list; } release inode lock; } Algorytm przydzialu nowego INODE ialloc() input: file system output: locked inode { while (not done) { if (superblock locked) { sleep (event superblock becomes free); continue; /* loop */ } if (inode list in superblock is empty) { lock superblock; get remembered inode for free inode search; search disk for free inodes until superblock full, or no more free inodes (algorithms bread and brelse); unlock superblock; wakeup (event superblock becomes free); if (no free inodes found on disk) return (no inode); set remembered inode for next free inode search; } /*there are inodes in superblock inode list */ iget(inode number from superblock inode list); if (inode not free after all) { write inode to disk; iput(inode); continue; /* loop */ } /*inode is free */ initialize inode; write inode to disk; decrement file system free inode count; return (inode); } } Algorytm zwalniania INODE ifree( ) input: file system inode number output: none { increment file system free inode count; if (superblock locked) return; if (inode list full) { if (inode number less than remembered inode for search) remembered inode for search = input inode number; } else store inode number in inode list; return; } Algorytm przydzialu bloku dyskowego alloc() input: file system number output: buffer for new block { while (superblock locked) sleep (event superblock not locked); remove block from superblock free list; if (removed last block from free list) { lock superblock; bread (block just taken from free list); copy block numbers in block into superblock; brelse(block buffer); unlock superblock; wakeup(event superblock not locked); } getblk (block removed from superblock list); zero buffer contents; decrement total count of free blocks; mark superblock modified; return buffer; } bmap () input: output: inode byte offset block number in file system byte offset into block bytes of I/O in block read ahead block number { calculate logical block number in file from byte offset; calculate start byte in block for I/O; /* output 2*/ calculate number of bytes to copy to user; /* output 3 */ check if read-ahead applicable, mark inode; /* output 4 */ determine level of block indirection; while (not at necessary level of indirection) { calculate index into inode or indirect block from logical block number in file; get disk block number from inode or indirect block; brelse(buffer from previous disk read, if any); if (no more levels of indirection) return (block number); bread (indirect block); adjust logical block number in file according to level of indirection; } } namei( ) input: path name output: locked inode { if (path name starts from root) wd = iget(root); else wd = iget(current directory inode); while (there is more path name) { read next path name component from input; verify wd is a directory, access permissions OK; if (wd is root && component is "..") continue; /* loop */ read wd using algorithms bmap, bread and brelse; if (component matches an entry in wd) { get inode number for matched component; iput(wd); wd = iget(matched component); } else /* component not in directory */ / return (no inode); } return (wd); } Algorytm tworzenia nowego pliku creat( ) input: output: file name permission settings file descriptor { namei(file name); if (file already exists) { if (not permitted access) { iput(inode); return (error); } } else /* file does not exist yet */ { ialloc(); create new directory entry in parent directory; include new file name and newly assigned inode number; } allocate file table entry for inode initialize count; if (file did exist at time of create) free all file blocks (algorithm free); unlock (inode); return(user file descriptor); } Algorytm otwierania pliku open() inputs: output: filename type of open file permissions (for creation type of open) file handle { namei(filename); if (file does not exist or not permitted access) return(error); allocate file table entry for inode, initialize count, offset; allocate user file descriptor entry, set pointer to file table entry; if (type of open specifies truncate file) free(all file blocks); unlock(inode); /* locked in namei */ return(user file descriptor); } Algorytmy czvtania i zapisu do pliku read/write() input: output: handle address number of bytes number of bytes read/written { get file table entry from user file descriptor; check file accessibility; set params in u-area: address, byte count, l/O direction; get inode from file table; lock inode; set byte offset in u-area from file table offset; while (count not satisfied) { disk block = bmap(file offset); calculate offset into block, number of bytes to read; ˝ ˝ /*==== read write differ in this part =====*/ /* version for read */ if (number of bytes == 0) break; /* end of file */ bread(disk block) /*breada if with read ahead*/ copy data to user address; /* version for write */ if(new block) { alloc(); /*multiple allocations if indirect block*/ ’ if (failure in allocation) break; /*out of disk space*/ } else if(writing part of block) bread(disk block); else getblk(); copy data to buffer; mark buffer as updated; /*===== algorithms similar from this point ====*/ update u-area fields: file byte offset, read count, user address; brelse(buffer); /* locked in bread */ } unlock(inode); update file table offset for next read; return (total number of bytes read/written); } Algorytm otwierania “pipe” bez nazwy pipe( ) input: none output: read file descriptor { write file descriptor; ialloc(new inode); allocate file table entry for reading, another for writing; initialize file table entries to point to new inode; allocate user file descriptor for reading; initialize to point to respective file table entries; set inode reference count to 2; initialize count of inode readers, writers to 1; } Algorytm tworzenia nowej pozycji katalogowej mknod( ) inputs: file name file type permissions major, minor device no. (block or character special files) output: none { if (new node not named pipe and user not super user) return(error); namei(file name); if (new node already exists) { iput(parent inode); return (error); } ialloc(); create new directory entry in parent directory: include new node name and newly assigned inode number; iput(parent inode); if (new node is block or character special file) write major, minor numbers into inode structure; iput(new node inode); } ! mount( ) inputs: output: file name of block special file directory name of mount point options (read only) none { if (not super user) return(error); get inode for block special file (algorithm namei); make legality checks; get inode for “mounted on” directory name (algorithm namei); if (not directory, or reference count > 1) { release inodes (algorithm iput); return (error); } find empty slot in mount table; invoke block device driver open routine; get free buffer from buffer cache; read superblock into free buffer; initialize superblock fields; get root inode of mounted device (algorithm iget), save in mount table; mark inode of “mounted on” directory as mount point; release special file inode (algorithm iput); unlock inode of mount point directory; } ! umount() input: special file name of file system to be unmounted output: none { if (not superuser) return(error); get inode of special file (algorithm namei); extract major, minor number of device being unmounted; get mount table entry, based on major, minor number; release inode of special file (algorithm iput); remove shared text entries from region table for files belonging to file system; update superblock, inodes, flush buffers; if (files from file system still in use) return (error); get root inode of mounted file system from mount table; lock inode; release inode (algorithm iput); /* iget was in mount */ invoke close routine for special device; invalidate buffers in pool from unmounted file system; get inode of mount point from mount table; lock inode; clear flag marking it as mount point; release inode (algorithm iput); /* iget in mount */ free buffer used for superblock; free mount table slot; } Zmodyfikowany algorytm iget (dla wielu systemów plików) iget( ) input: file system inode number output: locked inode { while (not done) { if (inode in inode cache) { if (inode locked) { sleep (event inode becomes unlocked); continue; /* loop */ } /* special processing for mount points */ if (inode a mount point) { find mount table entry for mount point; get new file system number from mount table; use root inode number in search; continue; * loop again */ } if (inode on inode free list) remove from free list; increment inode reference count; return (inode); } /* inode not in inode cache */ remove new inode from free list; reset inode number and file system; remove inode from old hash queue, place on new one; read inode from disk (algorithm bread); initialize inode (e.g. reference count to 1); return inode; } } Zmodyfikowany algorytm namei int namei() input: path name output: locked inode { if (path name starts from root) working inode = root inode (algorithm iget); else working inode = current directory inode (algorithm iget); while (there is more path name) { read next path name component from input; verify that inode is of directory, permissions; if (inode is of changed root and component is “..“) continue; /* loop */ component search: read inode (directory) (algorithms bmap, bread, brelse); if (component matches a directory entry) { get inode number for matched component; if (found inode of root and working inode is root and and component name is “..“) { /* crossing mount point */ get mount table entry for working inode; release working inode (algorithm iput); working inode = mounted on inode; lock mounted on inode; increment reference count of working inode; go to component search; /* for “..“ */ } iput (working inode); working inode = iget (new inode number); } else /* component not in directory */ return (no inode); } return (working inode); } link( ) input: output: existing file name new file name none { get inode for existing file name (algorith namei); if (too many links on file or linking directory without superuser permission) { release inode (algorithm iput); return(error); } increment link count on inode; update disk copy of inode; unlock inode; get parent inode for directory to contain new file name (algorith namei); if (new file name already exists or new file on different file system) { undo update done above; return(error); } create new directory entry in parent directory of new file name: include new file name, inode number of existing file name; release parent directory inode (algorith iput); release inode of existing file (algorith iput); } " unlink( ) input: file name output: none { get parent inode of file to be unlinked (algorithm namei); if (last component of file name is “.“) increment inode reference count; else get inode of file to be unlinked (algorithm iget); if (file is directory but user is not superuser) { release inodes (algorithm iput) ; return(error); } if (shared text file and link count currently 1) remove from region table; write parent directory: zero inode number of unlinked file; release inode parent directory (algorithm iput); decrement file link count; release file inode (algorithm iput); /* iput checks if link count is 0: if so, * releases file blocks (algorithm free) and * frees inode (algorithm ifree); */ }