resize.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:15k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  *   Copyright (c) International Business Machines  Corp., 2000-2002
  3.  *
  4.  *   This program is free software;  you can redistribute it and/or modify
  5.  *   it under the terms of the GNU General Public License as published by
  6.  *   the Free Software Foundation; either version 2 of the License, or 
  7.  *   (at your option) any later version.
  8.  * 
  9.  *   This program is distributed in the hope that it will be useful,
  10.  *   but WITHOUT ANY WARRANTY;  without even the implied warranty of
  11.  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
  12.  *   the GNU General Public License for more details.
  13.  *
  14.  *   You should have received a copy of the GNU General Public License
  15.  *   along with this program;  if not, write to the Free Software 
  16.  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  17. */
  18. #include <linux/fs.h>
  19. #include <linux/locks.h>
  20. #include "jfs_incore.h"
  21. #include "jfs_filsys.h"
  22. #include "jfs_metapage.h"
  23. #include "jfs_dinode.h"
  24. #include "jfs_imap.h"
  25. #include "jfs_dmap.h"
  26. #include "jfs_superblock.h"
  27. #include "jfs_txnmgr.h"
  28. #include "jfs_debug.h"
  29. #define BITSPERPAGE     (PSIZE << 3)
  30. #define L2MEGABYTE      20
  31. #define MEGABYTE        (1 << L2MEGABYTE)
  32. #define MEGABYTE32     (MEGABYTE << 5)
  33. /* convert block number to bmap file page number */
  34. #define BLKTODMAPN(b)
  35.         (((b) >> 13) + ((b) >> 23) + ((b) >> 33) + 3 + 1)
  36. /*
  37.  *      jfs_extendfs()
  38.  *
  39.  * function: extend file system;
  40.  *
  41.  *   |-------------------------------|----------|----------|
  42.  *   file system space               fsck       inline log
  43.  *                                   workspace  space
  44.  *
  45.  * input:
  46.  *      new LVSize: in LV blocks (required)
  47.  *      new LogSize: in LV blocks (optional)
  48.  *      new FSSize: in LV blocks (optional)
  49.  *
  50.  * new configuration:
  51.  * 1. set new LogSize as specified or default from new LVSize;
  52.  * 2. compute new FSCKSize from new LVSize;
  53.  * 3. set new FSSize as MIN(FSSize, LVSize-(LogSize+FSCKSize)) where
  54.  *    assert(new FSSize >= old FSSize),
  55.  *    i.e., file system must not be shrinked;
  56.  */
  57. int jfs_extendfs(struct super_block *sb, s64 newLVSize, int newLogSize)
  58. {
  59. int rc = 0;
  60. struct jfs_sb_info *sbi = JFS_SBI(sb);
  61. struct inode *ipbmap = sbi->ipbmap;
  62. struct inode *ipbmap2;
  63. struct inode *ipimap = sbi->ipimap;
  64. struct jfs_log *log = sbi->log;
  65. struct bmap *bmp = sbi->bmap;
  66. s64 newLogAddress, newFSCKAddress;
  67. int newFSCKSize;
  68. s64 newMapSize = 0, mapSize;
  69. s64 XAddress, XSize, nblocks, xoff, xaddr, t64;
  70. s64 oldLVSize;
  71. s64 newFSSize;
  72. s64 VolumeSize;
  73. int newNpages = 0, nPages, newPage, xlen, t32;
  74. int tid;
  75. int log_formatted = 0;
  76. struct inode *iplist[1];
  77. struct jfs_superblock *j_sb, *j_sb2;
  78. uint old_agsize;
  79. struct buffer_head *bh, *bh2;
  80. /* If the volume hasn't grown, get out now */
  81. if (sbi->mntflag & JFS_INLINELOG)
  82. oldLVSize = addressPXD(&sbi->logpxd) + lengthPXD(&sbi->logpxd);
  83. else
  84. oldLVSize = addressPXD(&sbi->fsckpxd) +
  85.     lengthPXD(&sbi->fsckpxd);
  86. if (oldLVSize >= newLVSize) {
  87. printk(KERN_WARNING
  88.        "jfs_extendfs: volume hasn't grown, returningn");
  89. goto out;
  90. }
  91. VolumeSize = sb->s_bdev->bd_inode->i_size >> sb->s_blocksize_bits;
  92. if (VolumeSize) {
  93. if (newLVSize > VolumeSize) {
  94. printk(KERN_WARNING "jfs_extendfs: invalid sizen");
  95. rc = -EINVAL;
  96. goto out;
  97. }
  98. } else {
  99. /* check the device */
  100. bh = sb_bread(sb, newLVSize - 1);
  101. if (!bh) {
  102. printk(KERN_WARNING "jfs_extendfs: invalid sizen");
  103. rc = -EINVAL;
  104. goto out;
  105. }
  106. bforget(bh);
  107. }
  108. /* Can't extend write-protected drive */
  109. if (isReadOnly(ipbmap)) {
  110. printk(KERN_WARNING "jfs_extendfs: read-only file systemn");
  111. rc = -EROFS;
  112. goto out;
  113. }
  114. /*
  115.  *      reconfigure LV spaces
  116.  *      ---------------------
  117.  *
  118.  * validate new size, or, if not specified, determine new size
  119.  */
  120. /*
  121.  * reconfigure inline log space:
  122.  */
  123. if ((sbi->mntflag & JFS_INLINELOG)) {
  124. if (newLogSize == 0) {
  125. /*
  126.  * no size specified: default to 1/256 of aggregate
  127.  * size; rounded up to a megabyte boundary;
  128.  */
  129. newLogSize = newLVSize >> 8;
  130. t32 = (1 << (20 - sbi->l2bsize)) - 1;
  131. newLogSize = (newLogSize + t32) & ~t32;
  132. newLogSize =
  133.     min(newLogSize, MEGABYTE32 >> sbi->l2bsize);
  134. } else {
  135. /*
  136.  * convert the newLogSize to fs blocks.
  137.  *
  138.  * Since this is given in megabytes, it will always be
  139.  * an even number of pages.
  140.  */
  141. newLogSize = (newLogSize * MEGABYTE) >> sbi->l2bsize;
  142. }
  143. } else
  144. newLogSize = 0;
  145. newLogAddress = newLVSize - newLogSize;
  146. /*
  147.  * reconfigure fsck work space:
  148.  *
  149.  * configure it to the end of the logical volume regardless of
  150.  * whether file system extends to the end of the aggregate;
  151.  * Need enough 4k pages to cover:
  152.  *  - 1 bit per block in aggregate rounded up to BPERDMAP boundary
  153.  *  - 1 extra page to handle control page and intermediate level pages
  154.  *  - 50 extra pages for the chkdsk service log
  155.  */
  156. t64 = ((newLVSize - newLogSize + BPERDMAP - 1) >> L2BPERDMAP)
  157.     << L2BPERDMAP;
  158. t32 = ((t64 + (BITSPERPAGE - 1)) / BITSPERPAGE) + 1 + 50;
  159. newFSCKSize = t32 << sbi->l2nbperpage;
  160. newFSCKAddress = newLogAddress - newFSCKSize;
  161. /*
  162.  * compute new file system space;
  163.  */
  164. newFSSize = newLVSize - newLogSize - newFSCKSize;
  165. /* file system cannot be shrinked */
  166. if (newFSSize < bmp->db_mapsize) {
  167. rc = EINVAL;
  168. goto out;
  169. }
  170. /*
  171.  * If we're expanding enough that the inline log does not overlap
  172.  * the old one, we can format the new log before we quiesce the
  173.  * filesystem.
  174.  */
  175. if ((sbi->mntflag & JFS_INLINELOG) && (newLogAddress > oldLVSize)) {
  176. if ((rc = lmLogFormat(log, newLogAddress, newLogSize)))
  177. goto out;
  178. log_formatted = 1;
  179. }
  180. /*
  181.  *      quiesce file system
  182.  *
  183.  * (prepare to move the inline log and to prevent map update)
  184.  *
  185.  * block any new transactions and wait for completion of
  186.  * all wip transactions and flush modified pages s.t.
  187.  * on-disk file system is in consistent state and
  188.  * log is not required for recovery.
  189.  */
  190. txQuiesce(sb);
  191. if (sbi->mntflag & JFS_INLINELOG) {
  192. /*
  193.  * deactivate old inline log
  194.  */
  195. lmLogShutdown(log);
  196. /*
  197.  * mark on-disk super block for fs in transition;
  198.  *
  199.  * update on-disk superblock for the new space configuration
  200.  * of inline log space and fsck work space descriptors:
  201.  * N.B. FS descriptor is NOT updated;
  202.  *
  203.  * crash recovery:
  204.  * logredo(): if FM_EXTENDFS, return to fsck() for cleanup;
  205.  * fsck(): if FM_EXTENDFS, reformat inline log and fsck
  206.  * workspace from superblock inline log descriptor and fsck
  207.  * workspace descriptor;
  208.  */
  209. /* read in superblock */
  210. if ((rc = readSuper(sb, &bh)))
  211. goto error_out;
  212. j_sb = (struct jfs_superblock *)bh->b_data;
  213. /* mark extendfs() in progress */
  214. j_sb->s_state |= cpu_to_le32(FM_EXTENDFS);
  215. j_sb->s_xsize = cpu_to_le64(newFSSize);
  216. PXDaddress(&j_sb->s_xfsckpxd, newFSCKAddress);
  217. PXDlength(&j_sb->s_xfsckpxd, newFSCKSize);
  218. PXDaddress(&j_sb->s_xlogpxd, newLogAddress);
  219. PXDlength(&j_sb->s_xlogpxd, newLogSize);
  220. /* synchronously update superblock */
  221. mark_buffer_dirty(bh);
  222. ll_rw_block(WRITE, 1, &bh);
  223. wait_on_buffer(bh);
  224. brelse(bh);
  225. /*
  226.  * format new inline log synchronously;
  227.  *
  228.  * crash recovery: if log move in progress,
  229.  * reformat log and exit success;
  230.  */
  231. if (!log_formatted)
  232. if ((rc = lmLogFormat(log, newLogAddress, newLogSize)))
  233. goto error_out;
  234. /*
  235.  * activate new log
  236.  */
  237. log->base = newLogAddress;
  238. log->size = newLogSize >> (L2LOGPSIZE - sb->s_blocksize_bits);
  239. if ((rc = lmLogInit(log)))
  240. goto error_out;
  241. }
  242. /*
  243.  *      extend block allocation map
  244.  *      ---------------------------
  245.  *
  246.  * extendfs() for new extension, retry after crash recovery;
  247.  *
  248.  * note: both logredo() and fsck() rebuild map from
  249.  * the bitmap and configuration parameter from superblock
  250.  * (disregarding all other control information in the map);
  251.  *
  252.  * superblock:
  253.  *  s_size: aggregate size in physical blocks;
  254.  */
  255. /*
  256.  *      compute the new block allocation map configuration
  257.  *
  258.  * map dinode:
  259.  *  di_size: map file size in byte;
  260.  *  di_nblocks: number of blocks allocated for map file;
  261.  *  di_mapsize: number of blocks in aggregate (covered by map);
  262.  * map control page:
  263.  *  db_mapsize: number of blocks in aggregate (covered by map);
  264.  */
  265. newMapSize = newFSSize;
  266. /* number of data pages of new bmap file:
  267.  * roundup new size to full dmap page boundary and
  268.  * add 1 extra dmap page for next extendfs()
  269.  */
  270. t64 = (newMapSize - 1) + BPERDMAP;
  271. newNpages = BLKTODMAPN(t64) + 1;
  272. /*
  273.  *      extend map from current map (WITHOUT growing mapfile)
  274.  *
  275.  * map new extension with unmapped part of the last partial
  276.  * dmap page, if applicable, and extra page(s) allocated
  277.  * at end of bmap by mkfs() or previous extendfs();
  278.  */
  279.       extendBmap:
  280. /* compute number of blocks requested to extend */
  281. mapSize = bmp->db_mapsize;
  282. XAddress = mapSize; /* eXtension Address */
  283. XSize = newMapSize - mapSize; /* eXtension Size */
  284. old_agsize = bmp->db_agsize; /* We need to know if this changes */
  285. /* compute number of blocks that can be extended by current mapfile */
  286. t64 = dbMapFileSizeToMapSize(ipbmap);
  287. if (mapSize > t64) {
  288. printk(KERN_ERR "jfs_extendfs: mapSize (0x%Lx) > t64 (0x%Lx)n",
  289.        (long long)mapSize, (long long)t64);
  290. rc = EIO;
  291. goto error_out;
  292. }
  293. nblocks = min(t64 - mapSize, XSize);
  294. /*
  295.  * update map pages for new extension:
  296.  *
  297.  * update/init dmap and bubble up the control hierarchy
  298.  * incrementally fold up dmaps into upper levels;
  299.  * update bmap control page;
  300.  */
  301. if ((rc = dbExtendFS(ipbmap, XAddress, nblocks)))
  302. goto error_out;
  303. /*
  304.  * the map now has extended to cover additional nblocks:
  305.  * dn_mapsize = oldMapsize + nblocks;
  306.  */
  307. /* ipbmap->i_mapsize += nblocks; */
  308. XSize -= nblocks;
  309. /*
  310.  *      grow map file to cover remaining extension
  311.  *      and/or one extra dmap page for next extendfs();
  312.  *
  313.  * allocate new map pages and its backing blocks, and
  314.  * update map file xtree
  315.  */
  316. /* compute number of data pages of current bmap file */
  317. nPages = ipbmap->i_size >> L2PSIZE;
  318. /* need to grow map file ? */
  319. if (nPages == newNpages)
  320. goto updateImap;
  321. /*
  322.  * grow bmap file for the new map pages required:
  323.  *
  324.  * allocate growth at the start of newly extended region;
  325.  * bmap file only grows sequentially, i.e., both data pages
  326.  * and possibly xtree index pages may grow in append mode,
  327.  * s.t. logredo() can reconstruct pre-extension state
  328.  * by washing away bmap file of pages outside s_size boundary;
  329.  */
  330. /*
  331.  * journal map file growth as if a regular file growth:
  332.  * (note: bmap is created with di_mode = IFJOURNAL|IFREG);
  333.  *
  334.  * journaling of bmap file growth is not required since
  335.  * logredo() do/can not use log records of bmap file growth
  336.  * but it provides careful write semantics, pmap update, etc.;
  337.  */
  338. /* synchronous write of data pages: bmap data pages are
  339.  * cached in meta-data cache, and not written out
  340.  * by txCommit();
  341.  */
  342. fsync_inode_data_buffers(ipbmap);
  343. diWriteSpecial(ipbmap, 0);
  344. newPage = nPages; /* first new page number */
  345. xoff = newPage << sbi->l2nbperpage;
  346. xlen = (newNpages - nPages) << sbi->l2nbperpage;
  347. xlen = min(xlen, (int) nblocks) & ~(sbi->nbperpage - 1);
  348. xaddr = XAddress;
  349. tid = txBegin(sb, COMMIT_FORCE);
  350. if ((rc = xtAppend(tid, ipbmap, 0, xoff, nblocks, &xlen, &xaddr, 0))) {
  351. txEnd(tid);
  352. goto error_out;
  353. }
  354. /* update bmap file size */
  355. ipbmap->i_size += xlen << sbi->l2bsize;
  356. ipbmap->i_blocks += LBLK2PBLK(sb, xlen);
  357. iplist[0] = ipbmap;
  358. rc = txCommit(tid, 1, &iplist[0], COMMIT_FORCE);
  359. txEnd(tid);
  360. if (rc)
  361. goto error_out;
  362. /*
  363.  * map file has been grown now to cover extension to further out;
  364.  * di_size = new map file size;
  365.  *
  366.  * if huge extension, the previous extension based on previous
  367.  * map file size may not have been sufficient to cover whole extension
  368.  * (it could have been used up for new map pages),
  369.  * but the newly grown map file now covers lot bigger new free space
  370.  * available for further extension of map;
  371.  */
  372. /* any more blocks to extend ? */
  373. if (XSize)
  374. goto extendBmap;
  375. /* finalize bmap */
  376. dbFinalizeBmap(ipbmap);
  377. /*
  378.  *      update inode allocation map
  379.  *      ---------------------------
  380.  *
  381.  * move iag lists from old to new iag;
  382.  * agstart field is not updated for logredo() to reconstruct
  383.  * iag lists if system crash occurs.
  384.  * (computation of ag number from agstart based on agsize
  385.  * will correctly identify the new ag);
  386.  */
  387.       updateImap:
  388. /* if new AG size the same as old AG size, done! */
  389. if (bmp->db_agsize != old_agsize) {
  390. if ((rc = diExtendFS(ipimap, ipbmap)))
  391. goto error_out;
  392. /* finalize imap */
  393. if ((rc = diSync(ipimap)))
  394. goto error_out;
  395. }
  396. /*
  397.  *      finalize
  398.  *      --------
  399.  *
  400.  * extension is committed when on-disk super block is
  401.  * updated with new descriptors: logredo will recover
  402.  * crash before it to pre-extension state;
  403.  */
  404. /* sync log to skip log replay of bmap file growth transaction; */
  405. /* lmLogSync(log, 1); */
  406. /*
  407.  * synchronous write bmap global control page;
  408.  * for crash before completion of write
  409.  * logredo() will recover to pre-extendfs state;
  410.  * for crash after completion of write,
  411.  * logredo() will recover post-extendfs state;
  412.  */
  413. if ((rc = dbSync(ipbmap)))
  414. goto error_out;
  415. /*
  416.  * copy primary bmap inode to secondary bmap inode
  417.  */
  418. ipbmap2 = diReadSpecial(sb, BMAP_I, 1);
  419. if (ipbmap2 == NULL) {
  420. printk(KERN_ERR "jfs_extendfs: diReadSpecial(bmap) failedn");
  421. goto error_out;
  422. }
  423. memcpy(&JFS_IP(ipbmap2)->i_xtroot, &JFS_IP(ipbmap)->i_xtroot, 288);
  424. ipbmap2->i_size = ipbmap->i_size;
  425. ipbmap2->i_blocks = ipbmap->i_blocks;
  426. diWriteSpecial(ipbmap2, 1);
  427. diFreeSpecial(ipbmap2);
  428. /*
  429.  *      update superblock
  430.  */
  431. if ((rc = readSuper(sb, &bh)))
  432. goto error_out;
  433. j_sb = (struct jfs_superblock *)bh->b_data;
  434. /* mark extendfs() completion */
  435. j_sb->s_state &= cpu_to_le32(~FM_EXTENDFS);
  436. j_sb->s_size = cpu_to_le64(bmp->db_mapsize) <<
  437.        le16_to_cpu(j_sb->s_l2bfactor);
  438. j_sb->s_agsize = cpu_to_le32(bmp->db_agsize);
  439. /* update inline log space descriptor */
  440. if (sbi->mntflag & JFS_INLINELOG) {
  441. PXDaddress(&(j_sb->s_logpxd), newLogAddress);
  442. PXDlength(&(j_sb->s_logpxd), newLogSize);
  443. }
  444. /* record log's mount serial number */
  445. j_sb->s_logserial = cpu_to_le32(log->serial);
  446. /* update fsck work space descriptor */
  447. PXDaddress(&(j_sb->s_fsckpxd), newFSCKAddress);
  448. PXDlength(&(j_sb->s_fsckpxd), newFSCKSize);
  449. j_sb->s_fscklog = 1;
  450. /* sb->s_fsckloglen remains the same */
  451. /* Update secondary superblock */
  452. bh2 = sb_bread(sb, SUPER2_OFF >> sb->s_blocksize_bits);
  453. if (bh2) {
  454. j_sb2 = (struct jfs_superblock *)bh2->b_data;
  455. memcpy(j_sb2, j_sb, sizeof (struct jfs_superblock));
  456. mark_buffer_dirty(bh);
  457. ll_rw_block(WRITE, 1, &bh2);
  458. wait_on_buffer(bh2);
  459. brelse(bh);
  460. }
  461. /* write primary superblock */
  462. mark_buffer_dirty(bh);
  463. ll_rw_block(WRITE, 1, &bh);
  464. wait_on_buffer(bh);
  465. brelse(bh);
  466. goto resume;
  467.       error_out:
  468. updateSuper(sb, FM_DIRTY);
  469.       resume:
  470. /*
  471.  *      resume file system transactions
  472.  */
  473. txResume(sb);
  474.       out:
  475. return rc;
  476. }