Ignore:
File:
1 edited

Legend:

Unmodified
Added
Removed
  • uspace/app/mkmfs/mkmfs.c

    r8367d1d r7a46bfe  
    11/*
     2 * Copyright (c) 2010 Jiri Svoboda
    23 * Copyright (c) 2011 Maurizio Lombardi
    34 * All rights reserved.
     
    5556#define USED    1
    5657
    57 #define UPPER(n, size)          (((n) / (size)) + (((n) % (size)) != 0))
    58 #define NEXT_DENTRY(p, dirsize) (p += (dirsize))
     58#define UPPER(n, size)                  (((n) / (size)) + (((n) % (size)) != 0))
     59#define NEXT_DENTRY(p, dirsize)         (p += (dirsize))
    5960
    6061typedef enum {
     
    6364} help_level_t;
    6465
    65 /* Generic MFS superblock */
     66/*Generic MFS superblock*/
    6667struct mfs_sb_info {
    6768        uint64_t n_inodes;
     
    8384
    8485static void     help_cmd_mkmfs(help_level_t level);
    85 static bool     is_power_of_two(uint32_t n);
     86static int      num_of_set_bits(uint32_t n);
    8687static int      init_superblock(struct mfs_sb_info *sb);
    8788static int      write_superblock(const struct mfs_sb_info *sbi);
     
    113114        int rc, c, opt_ind;
    114115        char *device_name;
    115         size_t devblock_size;
     116        aoff64_t devblock_size;
    116117
    117118        struct mfs_sb_info sb;
    118119
    119         /* Default is MinixFS V3 */
     120        /*Default is MinixFS V3*/
    120121        sb.magic = MFS_MAGIC_V3;
    121122        sb.fs_version = 3;
    122123
    123         /* Default block size is 4Kb */
     124        /*Default block size is 4Kb*/
    124125        sb.block_size = MFS_MAX_BLOCKSIZE;
    125126        sb.dirsize = MFS3_DIRSIZE;
     
    135136
    136137        for (c = 0, optind = 0, opt_ind = 0; c != -1;) {
    137                 c = getopt_long(argc, argv, "lh12b:i:",
    138                     long_options, &opt_ind);
     138                c = getopt_long(argc, argv, "lh12b:i:", long_options, &opt_ind);
    139139                switch (c) {
    140140                case 'h':
     
    169169
    170170        if (sb.block_size < MFS_MIN_BLOCKSIZE ||
    171             sb.block_size > MFS_MAX_BLOCKSIZE) {
     171                        sb.block_size > MFS_MAX_BLOCKSIZE) {
    172172                printf(NAME ":Error! Invalid block size.\n");
    173173                exit(0);
    174         } else if (!is_power_of_two(sb.block_size)) {
    175                 /* Block size must be a power of 2. */
     174        } else if (num_of_set_bits(sb.block_size) != 1) {
     175                /*Block size must be a power of 2.*/
    176176                printf(NAME ":Error! Invalid block size.\n");
    177177                exit(0);
    178178        } else if (sb.block_size > MFS_BLOCKSIZE &&
    179             sb.fs_version != 3) {
    180                 printf(NAME ":Error! Block size > 1024 is "
    181                     "supported by V3 filesystem only.\n");
     179                        sb.fs_version != 3) {
     180                printf(NAME ":Error! Block size > 1024 is supported by V3 filesystem only.\n");
    182181                exit(0);
    183182        } else if (sb.fs_version == 3 && sb.longnames) {
    184                 printf(NAME ":Error! Long filenames are supported "
    185                     "by V1/V2 filesystem only.\n");
     183                printf(NAME ":Error! Long filenames are supported by V1/V2 filesystem only.\n");
    186184                exit(0);
    187185        }
     
    223221        rc = block_get_nblocks(service_id, &sb.dev_nblocks);
    224222        if (rc != EOK) {
    225                 printf(NAME ": Warning, failed to obtain "
    226                     "block device size.\n");
     223                printf(NAME ": Warning, failed to obtain block device size.\n");
    227224        } else {
    228225                printf(NAME ": Block device has %" PRIuOFF64 " blocks.\n",
    229                     sb.dev_nblocks);
     226                                sb.dev_nblocks);
    230227        }
    231228
     
    235232        }
    236233
    237         /* Minimum block size is 1 Kb */
     234        /*Minimum block size is 1 Kb*/
    238235        sb.dev_nblocks /= 2;
    239236
    240237        printf(NAME ": Creating Minix file system on device\n");
    241         printf(NAME ": Writing superblock\n");
    242 
    243         /* Initialize superblock */
     238
     239        /*Initialize superblock*/
    244240        if (init_superblock(&sb) != EOK) {
    245241                printf(NAME ": Error. Superblock initialization failed\n");
     
    247243        }
    248244
    249         printf(NAME ": Initializing bitmaps\n");
    250 
    251         /* Initialize bitmaps */
     245        /*Initialize bitmaps*/
    252246        if (init_bitmaps(&sb) != EOK) {
    253247                printf(NAME ": Error. Bitmaps initialization failed\n");
     
    255249        }
    256250
    257         printf(NAME ": Initializing the inode table\n");
    258 
    259         /* Init inode table */
     251        /*Init inode table*/
    260252        if (init_inode_table(&sb) != EOK) {
    261253                printf(NAME ": Error. Inode table initialization failed\n");
     
    263255        }
    264256
    265         printf(NAME ": Creating the root directory inode\n");
    266 
    267         /* Make the root inode */
     257        /*Make the root inode*/
    268258        if (sb.fs_version == 1)
    269259                rc = make_root_ino(&sb);
     
    276266        }
    277267
    278         /* Insert directory entries . and .. */
     268        /*Insert directory entries . and ..*/
    279269        if (insert_dentries(&sb) != EOK) {
    280270                printf(NAME ": Error. Root directory initialization failed\n");
     
    309299
    310300        if (sb->fs_version != 3) {
    311                 /* Directory entries for V1/V2 filesystem */
     301                /*Directory entries for V1/V2 filesystem*/
    312302                struct mfs_dentry *dentry = root_block;
    313303
     
    316306
    317307                dentry = (struct mfs_dentry *) NEXT_DENTRY(dentry_ptr,
    318                     sb->dirsize);
     308                                sb->dirsize);
    319309
    320310                dentry->d_inum = MFS_ROOT_INO;
    321311                memcpy(dentry->d_name, "..\0", 3);
    322312        } else {
    323                 /* Directory entries for V3 filesystem */
     313                /*Directory entries for V3 filesystem*/
    324314                struct mfs3_dentry *dentry = root_block;
    325315
     
    328318
    329319                dentry = (struct mfs3_dentry *) NEXT_DENTRY(dentry_ptr,
    330                     sb->dirsize);
     320                                sb->dirsize);
    331321
    332322                dentry->d_inum = MFS_ROOT_INO;
     
    399389        ino_buf[MFS_ROOT_INO - 1].i_gid = 0;
    400390        ino_buf[MFS_ROOT_INO - 1].i_size = (sb->longnames ? MFSL_DIRSIZE :
    401             MFS_DIRSIZE) * 2;
     391        MFS_DIRSIZE) * 2;
    402392        ino_buf[MFS_ROOT_INO - 1].i_mtime = sec;
    403393        ino_buf[MFS_ROOT_INO - 1].i_nlinks = 2;
     
    421411        int rc;
    422412
    423         /* Compute offset of the first inode table block */
     413        /*Compute offset of the first inode table block*/
    424414        const long itable_off = sb->zbmap_blocks + sb->ibmap_blocks + 2;
    425415
     
    464454
    465455        if (sb->longnames)
    466                 sb->magic = sb->fs_version == 1 ? MFS_MAGIC_V1L :
    467                     MFS_MAGIC_V2L;
    468 
    469         /* Compute the number of zones on disk */
     456                sb->magic = sb->fs_version == 1 ? MFS_MAGIC_V1L : MFS_MAGIC_V2L;
     457
     458        /*Compute the number of zones on disk*/
    470459
    471460        if (sb->fs_version == 1) {
    472                 /* Valid only for MFS V1 */
     461                /*Valid only for MFS V1*/
    473462                sb->n_zones = sb->dev_nblocks > UINT16_MAX ?
    474                     UINT16_MAX : sb->dev_nblocks;
     463                                UINT16_MAX : sb->dev_nblocks;
    475464                ind = MFS_BLOCKSIZE / sizeof(uint16_t);
    476465                ind2 = ind * ind;
    477                 sb->max_file_size = (V1_NR_DIRECT_ZONES + ind + ind2) *
    478                     MFS_BLOCKSIZE;
     466                sb->max_file_size = (V1_NR_DIRECT_ZONES + ind + ind2) * MFS_BLOCKSIZE;
    479467        } else {
    480468                /*Valid for MFS V2/V3*/
     
    484472                else
    485473                        ptrsize = sizeof(uint32_t);
    486 
    487474                ind = sb->block_size / ptrsize;
    488475                ind2 = ind * ind;
     
    490477                sb->max_file_size = zones * sb->block_size;
    491478                sb->n_zones = sb->dev_nblocks > UINT32_MAX ?
    492                     UINT32_MAX : sb->dev_nblocks;
     479                                UINT32_MAX : sb->dev_nblocks;
    493480
    494481                if (sb->fs_version == 3) {
     
    500487        }
    501488
    502         /* Round up the number of inodes to fill block size */
     489        /*Round up the number of inodes to fill block size*/
    503490        if (sb->n_inodes == 0)
    504491                inodes = sb->dev_nblocks / 3;
     
    507494
    508495        if (inodes % sb->ino_per_block)
    509                 inodes = ((inodes / sb->ino_per_block) + 1) *
    510                     sb->ino_per_block;
     496                inodes = ((inodes / sb->ino_per_block) + 1) * sb->ino_per_block;
    511497
    512498        if (sb->fs_version < 3)
     
    515501                sb->n_inodes = inodes > UINT32_MAX ? UINT32_MAX : inodes;
    516502
    517         /* Compute inode bitmap size in blocks */
     503        /*Compute inode bitmap size in blocks*/
    518504        sb->ibmap_blocks = UPPER(sb->n_inodes, sb->block_size * 8);
    519505
    520         /* Compute inode table size */
     506        /*Compute inode table size*/
    521507        sb->itable_size = sb->n_inodes / sb->ino_per_block;
    522508
    523         /* Compute zone bitmap size in blocks */
     509        /*Compute zone bitmap size in blocks*/
    524510        sb->zbmap_blocks = UPPER(sb->n_zones, sb->block_size * 8);
    525511
    526         /* Compute first data zone position */
     512        /*Compute first data zone position*/
    527513        sb->first_data_zone = 2 + sb->itable_size +
    528             sb->zbmap_blocks + sb->ibmap_blocks;
    529 
    530         /* Set log2 of zone to block ratio to zero */
     514                        sb->zbmap_blocks + sb->ibmap_blocks;
     515
     516        /*Set log2 of zone to block ratio to zero*/
    531517        sb->log2_zone_size = 0;
    532518
    533         /* Check for errors */
     519        /*Check for errors*/
    534520        if (sb->first_data_zone >= sb->n_zones) {
    535521                printf(NAME ": Error! Insufficient disk space");
     
    537523        }
    538524
    539         /* Superblock is now ready to be written on disk */
     525        /*Superblock is now ready to be written on disk*/
    540526        printf(NAME ": %d block size\n", sb->block_size);
    541527        printf(NAME ": %d inodes\n", (uint32_t) sb->n_inodes);
     
    544530        printf(NAME ": inode bitmap blocks = %ld\n", sb->ibmap_blocks);
    545531        printf(NAME ": zone bitmap blocks = %ld\n", sb->zbmap_blocks);
    546         printf(NAME ": first data zone = %d\n", (uint32_t)sb->first_data_zone);
     532        printf(NAME ": first data zone = %d\n", (uint32_t) sb->first_data_zone);
    547533        printf(NAME ": max file size = %u\n", sb->max_file_size);
    548534        printf(NAME ": long fnames = %s\n", sb->longnames ? "Yes" : "No");
     
    659645        for (i = 0; i < ibmap_nblocks; ++i) {
    660646                if ((rc = write_block(start_block + i,
    661                     1, (ibmap_buf8 + i * sb->block_size))) != EOK)
     647                                1, (ibmap_buf8 + i * sb->block_size))) != EOK)
    662648                        return rc;
    663649        }
     
    667653        for (i = 0; i < zbmap_nblocks; ++i) {
    668654                if ((rc = write_block(start_block + i,
    669                     1, (zbmap_buf8 + i * sb->block_size))) != EOK)
     655                                1, (zbmap_buf8 + i * sb->block_size))) != EOK)
    670656                        return rc;
    671657        }
     
    706692                uint8_t *data_ptr = (uint8_t *) data;
    707693
    708                 rc = block_write_direct(service_id, tmp_off << 2,
    709                     size << 2, data_ptr);
     694                rc = block_write_direct(service_id, tmp_off << 2, size << 2, data_ptr);
    710695
    711696                if (rc != EOK)
     
    715700                tmp_off++;
    716701
    717                 return block_write_direct(service_id, tmp_off << 2,
    718                     size << 2, data_ptr);
    719         }
    720         return block_write_direct(service_id, off << shift,
    721             size << shift, data);
     702                return block_write_direct(service_id, tmp_off << 2, size << 2, data_ptr);
     703        }
     704        return block_write_direct(service_id, off << shift, size << shift, data);
    722705}
    723706
     
    728711        } else {
    729712                printf("Usage: [options] device\n"
    730                     "-1         Make a Minix version 1 filesystem\n"
    731                     "-2         Make a Minix version 2 filesystem\n"
    732                     "-b ##      Specify the block size in bytes (V3 only),\n"
    733                     "           valid block size values are 1024, 2048 and"
    734                                 " 4096 bytes per block\n"
    735                     "-i ##      Specify the number of inodes"
    736                                 " for the filesystem\n"
    737                     "-l         Use 30-char long filenames (V1/V2 only)\n");
    738         }
    739 }
    740 
    741 /** Check if a given number is a power of two.
    742  *
    743  * @param n     The number to check.
    744  *
    745  * @return      true if it is a power of two, false otherwise.
    746  */
    747 static bool is_power_of_two(uint32_t n)
    748 {
    749         if (n == 0)
    750                 return false;
    751 
    752         return (n & (n - 1)) == 0;
     713                                "-1         Make a Minix version 1 filesystem\n"
     714                                "-2         Make a Minix version 2 filesystem\n"
     715                                "-b ##      Specify the block size in bytes (V3 only),\n"
     716                                "           valid block size values are 1024, 2048 and 4096 bytes per block\n"
     717                                "-i ##      Specify the number of inodes for the filesystem\n"
     718                                "-l         Use 30-char long filenames (V1/V2 only)\n");
     719        }
     720}
     721
     722static int num_of_set_bits(uint32_t n)
     723{
     724        n = n - ((n >> 1) & 0x55555555);
     725        n = (n & 0x33333333) + ((n >> 2) & 0x33333333);
     726        return (((n + (n >> 4)) & 0xF0F0F0F) * 0x1010101) >> 24;
    753727}
    754728
Note: See TracChangeset for help on using the changeset viewer.