super.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 1,249 行 · 第 1/3 页

C
1,249
字号
	for (i = 0; i < blks; i += uspi->s_fpb) {		size = uspi->s_bsize;		if (i + uspi->s_fpb > blks)			size = (blks - i) * uspi->s_fsize;		if ((flags & UFS_TYPE_MASK) == UFS_TYPE_UFS2) {			ubh = ubh_bread(sb,				fs64_to_cpu(sb, usb->fs_u11.fs_u2.fs_csaddr) + i, size);			if (!ubh)				goto failed;			ubh_ubhcpymem (space, ubh, size);			sbi->s_csp[ufs_fragstoblks(i)]=(struct ufs_csum *)space;		}		else {			ubh = ubh_bread(sb, uspi->s_csaddr + i, size);			if (!ubh)				goto failed;			ubh_ubhcpymem(space, ubh, size);			sbi->s_csp[ufs_fragstoblks(i)]=(struct ufs_csum *)space;		}		space += size;		ubh_brelse (ubh);		ubh = NULL;	}	/*	 * Read cylinder group (we read only first fragment from block	 * at this time) and prepare internal data structures for cg caching.	 */	if (!(sbi->s_ucg = kmalloc (sizeof(struct buffer_head *) * uspi->s_ncg, GFP_KERNEL)))		goto failed;	for (i = 0; i < uspi->s_ncg; i++) 		sbi->s_ucg[i] = NULL;	for (i = 0; i < UFS_MAX_GROUP_LOADED; i++) {		sbi->s_ucpi[i] = NULL;		sbi->s_cgno[i] = UFS_CGNO_EMPTY;	}	for (i = 0; i < uspi->s_ncg; i++) {		UFSD(("read cg %u\n", i))		if (!(sbi->s_ucg[i] = sb_bread(sb, ufs_cgcmin(i))))			goto failed;		if (!ufs_cg_chkmagic (sb, (struct ufs_cylinder_group *) sbi->s_ucg[i]->b_data))			goto failed;#ifdef UFS_SUPER_DEBUG_MORE		ufs_print_cylinder_stuff(sb, (struct ufs_cylinder_group *) sbi->s_ucg[i]->b_data);#endif	}	for (i = 0; i < UFS_MAX_GROUP_LOADED; i++) {		if (!(sbi->s_ucpi[i] = kmalloc (sizeof(struct ufs_cg_private_info), GFP_KERNEL)))			goto failed;		sbi->s_cgno[i] = UFS_CGNO_EMPTY;	}	sbi->s_cg_loaded = 0;	UFSD(("EXIT\n"))	return 1;failed:	if (base) kfree (base);	if (sbi->s_ucg) {		for (i = 0; i < uspi->s_ncg; i++)			if (sbi->s_ucg[i]) brelse (sbi->s_ucg[i]);		kfree (sbi->s_ucg);		for (i = 0; i < UFS_MAX_GROUP_LOADED; i++)			if (sbi->s_ucpi[i]) kfree (sbi->s_ucpi[i]);	}	UFSD(("EXIT (FAILED)\n"))	return 0;}/* * Put on-disk structures associated with cylinder groups and  * write them back to disk */static void ufs_put_cylinder_structures (struct super_block *sb) {	struct ufs_sb_info * sbi = UFS_SB(sb);	struct ufs_sb_private_info * uspi;	struct ufs_buffer_head * ubh;	unsigned char * base, * space;	unsigned blks, size, i;		UFSD(("ENTER\n"))		uspi = sbi->s_uspi;	size = uspi->s_cssize;	blks = (size + uspi->s_fsize - 1) >> uspi->s_fshift;	base = space = (char*) sbi->s_csp[0];	for (i = 0; i < blks; i += uspi->s_fpb) {		size = uspi->s_bsize;		if (i + uspi->s_fpb > blks)			size = (blks - i) * uspi->s_fsize;		ubh = ubh_bread(sb, uspi->s_csaddr + i, size);		ubh_memcpyubh (ubh, space, size);		space += size;		ubh_mark_buffer_uptodate (ubh, 1);		ubh_mark_buffer_dirty (ubh);		ubh_brelse (ubh);	}	for (i = 0; i < sbi->s_cg_loaded; i++) {		ufs_put_cylinder (sb, i);		kfree (sbi->s_ucpi[i]);	}	for (; i < UFS_MAX_GROUP_LOADED; i++) 		kfree (sbi->s_ucpi[i]);	for (i = 0; i < uspi->s_ncg; i++) 		brelse (sbi->s_ucg[i]);	kfree (sbi->s_ucg);	kfree (base);	UFSD(("EXIT\n"))}static int ufs_fill_super(struct super_block *sb, void *data, int silent){	struct ufs_sb_info * sbi;	struct ufs_sb_private_info * uspi;	struct ufs_super_block_first * usb1;	struct ufs_super_block_second * usb2;	struct ufs_super_block_third * usb3;	struct ufs_super_block *usb;	struct ufs_buffer_head * ubh;		struct inode *inode;	unsigned block_size, super_block_size;	unsigned flags;	uspi = NULL;	ubh = NULL;	flags = 0;		UFSD(("ENTER\n"))			sbi = kmalloc(sizeof(struct ufs_sb_info), GFP_KERNEL);	if (!sbi)		goto failed_nomem;	sb->s_fs_info = sbi;	memset(sbi, 0, sizeof(struct ufs_sb_info));	UFSD(("flag %u\n", (int)(sb->s_flags & MS_RDONLY)))	#ifndef CONFIG_UFS_FS_WRITE	if (!(sb->s_flags & MS_RDONLY)) {		printk("ufs was compiled with read-only support, "		"can't be mounted as read-write\n");		goto failed;	}#endif	/*	 * Set default mount options	 * Parse mount options	 */	sbi->s_mount_opt = 0;	ufs_set_opt (sbi->s_mount_opt, ONERROR_LOCK);	if (!ufs_parse_options ((char *) data, &sbi->s_mount_opt)) {		printk("wrong mount options\n");		goto failed;	}	if (!(sbi->s_mount_opt & UFS_MOUNT_UFSTYPE)) {		if (!silent)			printk("You didn't specify the type of your ufs filesystem\n\n"			"mount -t ufs -o ufstype="			"sun|sunx86|44bsd|ufs2|5xbsd|old|hp|nextstep|netxstep-cd|openstep ...\n\n"			">>>WARNING<<< Wrong ufstype may corrupt your filesystem, "			"default is ufstype=old\n");		ufs_set_opt (sbi->s_mount_opt, UFSTYPE_OLD);	}	sbi->s_uspi = uspi =		kmalloc (sizeof(struct ufs_sb_private_info), GFP_KERNEL);	if (!uspi)		goto failed;	/* Keep 2Gig file limit. Some UFS variants need to override 	   this but as I don't know which I'll let those in the know loosen	   the rules */	   	switch (sbi->s_mount_opt & UFS_MOUNT_UFSTYPE) {	case UFS_MOUNT_UFSTYPE_44BSD:		UFSD(("ufstype=44bsd\n"))		uspi->s_fsize = block_size = 512;		uspi->s_fmask = ~(512 - 1);		uspi->s_fshift = 9;		uspi->s_sbsize = super_block_size = 1536;		uspi->s_sbbase = 0;		flags |= UFS_DE_44BSD | UFS_UID_44BSD | UFS_ST_44BSD | UFS_CG_44BSD;		break;	case UFS_MOUNT_UFSTYPE_UFS2:		UFSD(("ufstype=ufs2\n"))		uspi->s_fsize = block_size = 512;		uspi->s_fmask = ~(512 - 1);		uspi->s_fshift = 9;		uspi->s_sbsize = super_block_size = 1536;		uspi->s_sbbase =  0;		flags |= UFS_TYPE_UFS2 | UFS_DE_44BSD | UFS_UID_44BSD | UFS_ST_44BSD | UFS_CG_44BSD;		if (!(sb->s_flags & MS_RDONLY)) {			printk(KERN_INFO "ufstype=ufs2 is supported read-only\n");			sb->s_flags |= MS_RDONLY; 		}		break;			case UFS_MOUNT_UFSTYPE_SUN:		UFSD(("ufstype=sun\n"))		uspi->s_fsize = block_size = 1024;		uspi->s_fmask = ~(1024 - 1);		uspi->s_fshift = 10;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		uspi->s_maxsymlinklen = 56;		flags |= UFS_DE_OLD | UFS_UID_EFT | UFS_ST_SUN | UFS_CG_SUN;		break;	case UFS_MOUNT_UFSTYPE_SUNx86:		UFSD(("ufstype=sunx86\n"))		uspi->s_fsize = block_size = 1024;		uspi->s_fmask = ~(1024 - 1);		uspi->s_fshift = 10;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		uspi->s_maxsymlinklen = 56;		flags |= UFS_DE_OLD | UFS_UID_EFT | UFS_ST_SUNx86 | UFS_CG_SUN;		break;	case UFS_MOUNT_UFSTYPE_OLD:		UFSD(("ufstype=old\n"))		uspi->s_fsize = block_size = 1024;		uspi->s_fmask = ~(1024 - 1);		uspi->s_fshift = 10;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		flags |= UFS_DE_OLD | UFS_UID_OLD | UFS_ST_OLD | UFS_CG_OLD;		if (!(sb->s_flags & MS_RDONLY)) {			if (!silent)				printk(KERN_INFO "ufstype=old is supported read-only\n");			sb->s_flags |= MS_RDONLY;		}		break;		case UFS_MOUNT_UFSTYPE_NEXTSTEP:		UFSD(("ufstype=nextstep\n"))		uspi->s_fsize = block_size = 1024;		uspi->s_fmask = ~(1024 - 1);		uspi->s_fshift = 10;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		flags |= UFS_DE_OLD | UFS_UID_OLD | UFS_ST_OLD | UFS_CG_OLD;		if (!(sb->s_flags & MS_RDONLY)) {			if (!silent)				printk(KERN_INFO "ufstype=nextstep is supported read-only\n");			sb->s_flags |= MS_RDONLY;		}		break;		case UFS_MOUNT_UFSTYPE_NEXTSTEP_CD:		UFSD(("ufstype=nextstep-cd\n"))		uspi->s_fsize = block_size = 2048;		uspi->s_fmask = ~(2048 - 1);		uspi->s_fshift = 11;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		flags |= UFS_DE_OLD | UFS_UID_OLD | UFS_ST_OLD | UFS_CG_OLD;		if (!(sb->s_flags & MS_RDONLY)) {			if (!silent)				printk(KERN_INFO "ufstype=nextstep-cd is supported read-only\n");			sb->s_flags |= MS_RDONLY;		}		break;		case UFS_MOUNT_UFSTYPE_OPENSTEP:		UFSD(("ufstype=openstep\n"))		uspi->s_fsize = block_size = 1024;		uspi->s_fmask = ~(1024 - 1);		uspi->s_fshift = 10;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		flags |= UFS_DE_44BSD | UFS_UID_44BSD | UFS_ST_44BSD | UFS_CG_44BSD;		if (!(sb->s_flags & MS_RDONLY)) {			if (!silent)				printk(KERN_INFO "ufstype=openstep is supported read-only\n");			sb->s_flags |= MS_RDONLY;		}		break;		case UFS_MOUNT_UFSTYPE_HP:		UFSD(("ufstype=hp\n"))		uspi->s_fsize = block_size = 1024;		uspi->s_fmask = ~(1024 - 1);		uspi->s_fshift = 10;		uspi->s_sbsize = super_block_size = 2048;		uspi->s_sbbase = 0;		flags |= UFS_DE_OLD | UFS_UID_OLD | UFS_ST_OLD | UFS_CG_OLD;		if (!(sb->s_flags & MS_RDONLY)) {			if (!silent)				printk(KERN_INFO "ufstype=hp is supported read-only\n");			sb->s_flags |= MS_RDONLY; 		} 		break;	default:		if (!silent)			printk("unknown ufstype\n");		goto failed;	}	again:		if (!sb_set_blocksize(sb, block_size)) {		printk(KERN_ERR "UFS: failed to set blocksize\n");		goto failed;	}	/*	 * read ufs super block from device	 */	if ( (flags & UFS_TYPE_MASK) == UFS_TYPE_UFS2) {		ubh = ubh_bread_uspi(uspi, sb, uspi->s_sbbase + SBLOCK_UFS2/block_size, super_block_size);	}	else {		ubh = ubh_bread_uspi(uspi, sb, uspi->s_sbbase + UFS_SBLOCK/block_size, super_block_size);	}	if (!ubh)             goto failed;		usb1 = ubh_get_usb_first(USPI_UBH);	usb2 = ubh_get_usb_second(USPI_UBH);	usb3 = ubh_get_usb_third(USPI_UBH);	usb  = (struct ufs_super_block *)		((struct ufs_buffer_head *)uspi)->bh[0]->b_data ;	/*	 * Check ufs magic number	 */	sbi->s_bytesex = BYTESEX_LE;	switch ((uspi->fs_magic = fs32_to_cpu(sb, usb3->fs_magic))) {		case UFS_MAGIC:		case UFS2_MAGIC:		case UFS_MAGIC_LFN:	        case UFS_MAGIC_FEA:	        case UFS_MAGIC_4GB:			goto magic_found;	}	sbi->s_bytesex = BYTESEX_BE;	switch ((uspi->fs_magic = fs32_to_cpu(sb, usb3->fs_magic))) {		case UFS_MAGIC:		case UFS2_MAGIC:		case UFS_MAGIC_LFN:	        case UFS_MAGIC_FEA:	        case UFS_MAGIC_4GB:			goto magic_found;	}	if ((((sbi->s_mount_opt & UFS_MOUNT_UFSTYPE) == UFS_MOUNT_UFSTYPE_NEXTSTEP) 	  || ((sbi->s_mount_opt & UFS_MOUNT_UFSTYPE) == UFS_MOUNT_UFSTYPE_NEXTSTEP_CD) 	  || ((sbi->s_mount_opt & UFS_MOUNT_UFSTYPE) == UFS_MOUNT_UFSTYPE_OPENSTEP)) 	  && uspi->s_sbbase < 256) {		ubh_brelse_uspi(uspi);		ubh = NULL;		uspi->s_sbbase += 8;		goto again;	}	if (!silent)		printk("ufs_read_super: bad magic number\n");	goto failed;magic_found:	/*	 * Check block and fragment sizes	 */	uspi->s_bsize = fs32_to_cpu(sb, usb1->fs_bsize);	uspi->s_fsize = fs32_to_cpu(sb, usb1->fs_fsize);	uspi->s_sbsize = fs32_to_cpu(sb, usb1->fs_sbsize);	uspi->s_fmask = fs32_to_cpu(sb, usb1->fs_fmask);	uspi->s_fshift = fs32_to_cpu(sb, usb1->fs_fshift);	if (uspi->s_fsize & (uspi->s_fsize - 1)) {		printk(KERN_ERR "ufs_read_super: fragment size %u is not a power of 2\n",			uspi->s_fsize);			goto failed;	}	if (uspi->s_fsize < 512) {		printk(KERN_ERR "ufs_read_super: fragment size %u is too small\n",			uspi->s_fsize);		goto failed;	}	if (uspi->s_fsize > 4096) {		printk(KERN_ERR "ufs_read_super: fragment size %u is too large\n",			uspi->s_fsize);		goto failed;	}	if (uspi->s_bsize & (uspi->s_bsize - 1)) {		printk(KERN_ERR "ufs_read_super: block size %u is not a power of 2\n",			uspi->s_bsize);		goto failed;	}	if (uspi->s_bsize < 4096) {		printk(KERN_ERR "ufs_read_super: block size %u is too small\n",			uspi->s_bsize);		goto failed;	}	if (uspi->s_bsize / uspi->s_fsize > 8) {		printk(KERN_ERR "ufs_read_super: too many fragments per block (%u)\n",			uspi->s_bsize / uspi->s_fsize);		goto failed;	}	if (uspi->s_fsize != block_size || uspi->s_sbsize != super_block_size) {		ubh_brelse_uspi(uspi);		ubh = NULL;		block_size = uspi->s_fsize;		super_block_size = uspi->s_sbsize;		UFSD(("another value of block_size or super_block_size %u, %u\n", block_size, super_block_size))		goto again;	}#ifdef UFS_SUPER_DEBUG_MORE        if ((flags & UFS_TYPE_MASK) == UFS_TYPE_UFS2)		ufs2_print_super_stuff(sb,usb);        else		ufs_print_super_stuff(sb, usb1, usb2, usb3);#endif	/*

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?