真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

基于linux分怎么實(shí)現(xiàn)虛擬文件系統(tǒng)初始化

本篇內(nèi)容主要講解“基于linux分怎么實(shí)現(xiàn)虛擬文件系統(tǒng)初始化”,感興趣的朋友不妨來(lái)看看。本文介紹的方法操作簡(jiǎn)單快捷,實(shí)用性強(qiáng)。下面就讓小編來(lái)帶大家學(xué)習(xí)“基于linux分怎么實(shí)現(xiàn)虛擬文件系統(tǒng)初始化”吧!

洮南網(wǎng)站制作公司哪家好,找成都創(chuàng)新互聯(lián)公司!從網(wǎng)頁(yè)設(shè)計(jì)、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、自適應(yīng)網(wǎng)站建設(shè)等網(wǎng)站項(xiàng)目制作,到程序開發(fā),運(yùn)營(yíng)維護(hù)。成都創(chuàng)新互聯(lián)公司公司2013年成立到現(xiàn)在10年的時(shí)間,我們擁有了豐富的建站經(jīng)驗(yàn)和運(yùn)維經(jīng)驗(yàn),來(lái)保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選成都創(chuàng)新互聯(lián)公司。

從main函數(shù)開始,直到虛擬文件系統(tǒng)的初始化,路徑是init()->setup()->syssetup();sys_setup主要是注冊(cè)了虛擬文件系統(tǒng)下面所有的文件系統(tǒng)。然后掛載根文件系統(tǒng)。下面是初始化代碼。

   

asmlinkage int sys_setup(void)
{
static int callable = 1;

if (!callable)
return -1;
callable = 0;

device_setup();

#ifdef CONFIG_MINIX_FS
register_filesystem(&(struct file_system_type)
{minix_read_super, "minix", 1, NULL});
#endif

#ifdef CONFIG_EXT_FS
register_filesystem(&(struct file_system_type)
{ext_read_super, "ext", 1, NULL});
#endif
......
mount_root();
}              

下面先看一下基本的數(shù)據(jù)結(jié)構(gòu)。

   
     
 
    
   

struct file_system_type {
struct super_block *(*read_super) (struct super_block *, void *, int);
char *name;
int requires_dev;
struct file_system_type * next;
};              

這是一個(gè)具體文件系統(tǒng)在虛擬文件系統(tǒng)注冊(cè)時(shí)的表示結(jié)構(gòu)。然后看一下注冊(cè)文件系統(tǒng)的函數(shù)。

   
     
 
    
   

int register_filesystem(struct file_system_type * fs)
{
struct file_system_type ** tmp;

if (!fs)
return -EINVAL;
if (fs->next)
return -EBUSY;
// tmp是二級(jí)指針,指向文件系統(tǒng)鏈表的頭指針的地址
tmp = &file_systems;
// 遍歷鏈表,直到尾部,插入新的節(jié)點(diǎn)
while (*tmp) {
// 判斷是否已經(jīng)注冊(cè)了該文件系統(tǒng)
if (strcmp((*tmp)->name, fs->name) == 0)
return -EBUSY;
// 指向當(dāng)前節(jié)點(diǎn)的next域的地址,*tmp得到下一個(gè)被比較的節(jié)點(diǎn)
tmp = &(*tmp)->next;
}
// 利用二級(jí)指針指針修改next域的內(nèi)容,不需要使用->next = fs的形式
*tmp = fs;
return 0;
}              

就是把一個(gè)file_system_type結(jié)構(gòu)體插入一個(gè)鏈表中。注冊(cè)文件系統(tǒng)其實(shí)只是構(gòu)建一個(gè)單鏈表。接著看掛載根文件系統(tǒng)。這里大致分析一下流程。有時(shí)間再詳細(xì)說(shuō)。

   
     
 
    
   

void mount_root(void)
{
struct file_system_type * fs_type;
struct super_block * sb;
struct inode * inode, d_inode;
struct file filp;
int retval;
// 清空超級(jí)塊數(shù)組
memset(super_blocks, 0, sizeof(super_blocks));
#ifdef CONFIG_BLK_DEV_FD
if (MAJOR(ROOT_DEV) == FLOPPY_MAJOR) {
printk(KERN_NOTICE "VFS: Insert root floppy and press ENTER\n");
wait_for_keypress();
}
#endif

memset(&filp, 0, sizeof(filp));
memset(&d_inode, 0, sizeof(d_inode));
// 根設(shè)備號(hào)
d_inode.i_rdev = ROOT_DEV;
filp.f_inode = &d_inode;
// 只讀方式掛載
if ( root_mountflags & MS_RDONLY)
filp.f_mode = 1; /* read only */
else
filp.f_mode = 3; /* read write */
// 暫時(shí)忽略
retval = blkdev_open(&d_inode, &filp);
if(retval == -EROFS){
root_mountflags |= MS_RDONLY;
filp.f_mode = 1;
retval = blkdev_open(&d_inode, &filp);
}

for (fs_type = file_systems ; fs_type ; fs_type = fs_type->next) {
if(retval)
break;
// 沒(méi)有關(guān)聯(lián)到設(shè)備則不需要往下執(zhí)行,有些文件系統(tǒng)是沒(méi)有對(duì)應(yīng)的底層設(shè)備的
if (!fs_type->requires_dev)
continue;
// 讀根設(shè)備的超級(jí)塊,設(shè)備的第一扇區(qū)是分區(qū)表,接著是超級(jí)塊
sb = read_super(ROOT_DEV,fs_type->name,root_mountflags,NULL,1);
// 讀取成功
if (sb) {
// 根節(jié)點(diǎn)
inode = sb->s_mounted;
inode->i_count += 3 ; /* NOTE! it is logically used 4 times, not 1 */
sb->s_covered = inode;
sb->s_flags = root_mountflags;
// 當(dāng)前進(jìn)程(init進(jìn)程)的根目錄和工作目錄設(shè)置為根節(jié)點(diǎn)
current->fs->pwd = inode;
current->fs->root = inode;
printk ("VFS: Mounted root (%s filesystem)%s.\n",
fs_type->name,
(sb->s_flags & MS_RDONLY) ? " readonly" : "");
// 直接返回,即第一個(gè)讀取成功的文件系統(tǒng)成為根文件系統(tǒng)
return;
}
}
panic("VFS: Unable to mount root fs on %02x:%02x",
MAJOR(ROOT_DEV), MINOR(ROOT_DEV));
}              

讀取根設(shè)備的超級(jí)塊內(nèi)容,如果成功,則成為根文件系統(tǒng)。并設(shè)置當(dāng)前init進(jìn)程的工作目錄和根目錄是根文件系統(tǒng)的根節(jié)點(diǎn)對(duì)應(yīng)的inode。我們看看怎么讀取超級(jí)塊的。

   
     
 
    
   

// 讀設(shè)備對(duì)應(yīng)的超級(jí)塊
static struct super_block * read_super(dev_t dev,char *name,int flags,
  void *data, int silent)
{
struct super_block * s;
struct file_system_type *type;

if (!dev)
return NULL;
check_disk_change(dev);
// 有則直接返回,初始化的時(shí)候還沒(méi)有
s = get_super(dev);
if (s)
return s;
// 否則根據(jù)name在文件系統(tǒng)鏈表中(在系統(tǒng)初始化時(shí)建立的)找到對(duì)應(yīng)的文件系統(tǒng)節(jié)點(diǎn),里面有一個(gè)read_super函數(shù)
if (!(type = get_fs_type(name))) {
printk("VFS: on device %d/%d: get_fs_type(%s) failed\n",
MAJOR(dev), MINOR(dev), name);
return NULL;
}
// 在超級(jí)塊數(shù)組中找到一個(gè)slot
for (s = 0+super_blocks ;; s++) {
if (s >= NR_SUPER+super_blocks)
return NULL;
if (!s->s_dev)
break;
}
// 賦值給超級(jí)塊節(jié)點(diǎn)的字段
s->s_dev = dev;
s->s_flags = flags;
// 調(diào)底層的文件系統(tǒng)到硬盤去讀取超級(jí)塊內(nèi)容,比如ext文件系統(tǒng),ext2文件系統(tǒng)等等都定義了該函數(shù)。
if (!type->read_super(s,data, silent)) {
s->s_dev = 0;
return NULL;
}
s->s_dev = dev;
s->s_covered = NULL;
s->s_rd_only = 0;
s->s_dirt = 0;
s->s_type = type;
return s;
}              

主要是獲取一個(gè)超級(jí)塊結(jié)構(gòu)體,然后調(diào)底層文件系統(tǒng)的read_super。然后設(shè)置超級(jí)塊的屬性。這里分析一下ext文件系統(tǒng)的read_super。

   
     
 
    
   

struct super_block *ext_read_super(struct super_block *s,void *data,
  int silent)
{
struct buffer_head *bh;
struct ext_super_block *es;
int dev = s->s_dev,block;

lock_super(s);
set_blocksize(dev, BLOCK_SIZE);
// 讀取設(shè)備的內(nèi)容,即超級(jí)塊的內(nèi)容
if (!(bh = bread(dev, 1, BLOCK_SIZE))) {
s->s_dev=0;
unlock_super(s);
printk("EXT-fs: unable to read superblock\n");
return NULL;
}
// 文件系統(tǒng)的一些屬性
es = (struct ext_super_block *) bh->b_data;
s->s_blocksize = 1024;
s->s_blocksize_bits = 10;
s->u.ext_sb.s_ninodes = es->s_ninodes;
s->u.ext_sb.s_nzones = es->s_nzones;
s->u.ext_sb.s_firstdatazone = es->s_firstdatazone;
s->u.ext_sb.s_log_zone_size = es->s_log_zone_size;
s->u.ext_sb.s_max_size = es->s_max_size;
s->s_magic = es->s_magic;
s->u.ext_sb.s_firstfreeblocknumber = es->s_firstfreeblock;
s->u.ext_sb.s_freeblockscount = es->s_freeblockscount;
s->u.ext_sb.s_firstfreeinodenumber = es->s_firstfreeinode;
s->u.ext_sb.s_freeinodescount = es->s_freeinodescount;
brelse(bh);
if (s->s_magic != EXT_SUPER_MAGIC) {
s->s_dev = 0;
unlock_super(s);
if (!silent)
printk("VFS: Can't find an extfs filesystem on dev 0x%04x.\n",
  dev);
return NULL;
}
if (!s->u.ext_sb.s_firstfreeblocknumber)
s->u.ext_sb.s_firstfreeblock = NULL;
else
if (!(s->u.ext_sb.s_firstfreeblock = bread(dev,
s->u.ext_sb.s_firstfreeblocknumber, BLOCK_SIZE))) {
printk("ext_read_super: unable to read first free block\n");
s->s_dev = 0;
unlock_super(s);
return NULL;
}
if (!s->u.ext_sb.s_firstfreeinodenumber)
s->u.ext_sb.s_firstfreeinodeblock = NULL;
else {
block = 2 + (s->u.ext_sb.s_firstfreeinodenumber - 1) / EXT_INODES_PER_BLOCK;
if (!(s->u.ext_sb.s_firstfreeinodeblock = bread(dev, block, BLOCK_SIZE))) {
printk("ext_read_super: unable to read first free inode block\n");
brelse(s->u.ext_sb.s_firstfreeblock);
s->s_dev = 0;
unlock_super (s);
return NULL;
}
}
unlock_super(s);
/* set up enough so that it can read an inode */
s->s_dev = dev;
// 操作函數(shù)集
s->s_op = &ext_sops;
// 讀取根節(jié)點(diǎn)
if (!(s->s_mounted = iget(s,EXT_ROOT_INO))) {
s->s_dev=0;
printk("EXT-fs: get root inode failed\n");
return NULL;
}
return s;
}              

主要工作是讀取硬盤的超級(jí)塊信息到內(nèi)存,最后讀取根節(jié)點(diǎn)對(duì)應(yīng)的inode。

到此,相信大家對(duì)“基于linux分怎么實(shí)現(xiàn)虛擬文件系統(tǒng)初始化”有了更深的了解,不妨來(lái)實(shí)際操作一番吧!這里是創(chuàng)新互聯(lián)網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!


網(wǎng)頁(yè)題目:基于linux分怎么實(shí)現(xiàn)虛擬文件系統(tǒng)初始化
URL分享:http://weahome.cn/article/ihdddo.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部