10133 smatch fixes for usr/src/cmd/fs.d

   1 /*
   2  * CDDL HEADER START
   3  *
   4  * The contents of this file are subject to the terms of the
   5  * Common Development and Distribution License (the "License").
   6  * You may not use this file except in compliance with the License.
   7  *
   8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
   9  * or http://www.opensolaris.org/os/licensing.
  10  * See the License for the specific language governing permissions
  11  * and limitations under the License.
  12  *
  13  * When distributing Covered Code, include this CDDL HEADER in each
  14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
  15  * If applicable, add the following below this CDDL HEADER, with the
  16  * fields enclosed by brackets "[]" replaced with your own identifying
  17  * information: Portions Copyright [yyyy] [name of copyright owner]
  18  *
  19  * CDDL HEADER END
  20  */
  21 
  22 /*
  23  * Copyright (c) 1988, 2010, Oracle and/or its affiliates. All rights reserved.
  24  */
  25 
  26 /*      Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T     */
  27 /*      All Rights Reserved     */
  28 
  29 /*
  30  * University Copyright- Copyright (c) 1982, 1986, 1988
  31  * The Regents of the University of California
  32  * All Rights Reserved
  33  *
  34  * University Acknowledgment- Portions of this document are derived from
  35  * software developed by the University of California, Berkeley, and its
  36  * contributors.
  37  */
  38 
  39 /*




  40  * The maximum supported file system size (in sectors) is the
  41  * number of frags that can be represented in an int32_t field
  42  * (INT_MAX) times the maximum number of sectors per frag.  Since
  43  * the maximum frag size is MAXBSIZE, the maximum number of sectors
  44  * per frag is MAXBSIZE/DEV_BSIZE.
  45  */
  46 #define FS_MAX  (((diskaddr_t)INT_MAX) * (MAXBSIZE/DEV_BSIZE))
  47 
  48 /*
  49  * make file system for cylinder-group style file systems
  50  *
  51  * usage:
  52  *
  53  *    mkfs [-F FSType] [-V] [-G [-P]] [-M dirname] [-m] [options]
  54  *      [-o specific_options]  special size
  55  *      [nsect ntrack bsize fsize cpg   minfree rps nbpi opt apc rotdelay
  56  *        2     3      4     5     6    7       8   9    10  11  12
  57  *      nrpos maxcontig mtb]
  58  *      13    14        15
  59  *
  60  *  where specific_options are:
  61  *      N - no create
  62  *      nsect - The number of sectors per track
  63  *      ntrack - The number of tracks per cylinder
  64  *      bsize - block size
  65  *      fragsize - fragment size
  66  *      cgsize - The number of disk cylinders per cylinder group.
  67  *      free - minimum free space
  68  *      rps - rotational speed (rev/sec).
  69  *      nbpi - number of data bytes per allocated inode
  70  *      opt - optimization (space, time)
  71  *      apc - number of alternates
  72  *      gap - gap size
  73  *      nrpos - number of rotational positions
  74  *      maxcontig - maximum number of logical blocks that will be
  75  *              allocated contiguously before inserting rotational delay
  76  *      mtb - if "y", set up file system for eventual growth to over a
  77  *              a terabyte
  78  * -P Do not grow the file system, but print on stdout the maximal
  79  *    size in sectors to which the file system can be increased. The calculated
  80  *    size is limited by the value provided by the operand size.
  81  *
  82  * Note that -P is a project-private interface and together with -G intended
  83  * to be used only by the growfs script. It is therefore purposely not
  84  * documented in the man page.
  85  * The -P option is covered by PSARC case 2003/422.
  86  */
  87 
  88 /*
  89  * The following constants set the defaults used for the number
  90  * of sectors/track (fs_nsect), and number of tracks/cyl (fs_ntrak).
  91  *
  92  *                      NSECT           NTRAK
  93  *      72MB CDC        18              9
  94  *      30MB CDC        18              5
  95  *      720KB Diskette  9               2
  96  *
  97  * However the defaults will be different for disks larger than CHSLIMIT.
  98  */
  99 
 100 #define DFLNSECT        32
 101 #define DFLNTRAK        16
 102 
 103 /*
 104  * The following default sectors and tracks values are used for
 105  * non-efi disks that are larger than the CHS addressing limit. The
 106  * existing default cpg of 16 (DESCPG) holds good for larger disks too.
 107  */
 108 #define DEF_SECTORS_EFI 128
 109 #define DEF_TRACKS_EFI  48
 110 
 111 /*
 112  * The maximum number of cylinders in a group depends upon how much
 113  * information can be stored on a single cylinder. The default is to
 114  * use 16 cylinders per group.  This is effectively tradition - it was
 115  * the largest value acceptable under SunOs 4.1
 116  */
 117 #define DESCPG          16      /* desired fs_cpg */
 118 
 119 /*
 120  * The following two constants set the default block and fragment sizes.
 121  * Both constants must be a power of 2 and meet the following constraints:
 122  *      MINBSIZE <= DESBLKSIZE <= MAXBSIZE
 123  *      DEV_BSIZE <= DESFRAGSIZE <= DESBLKSIZE
 124  *      DESBLKSIZE / DESFRAGSIZE <= 8
 125  */
 126 #define DESBLKSIZE      8192
 127 #define DESFRAGSIZE     1024
 128 
 129 /*
 130  * MINFREE gives the minimum acceptable percentage of file system
 131  * blocks which may be free. If the freelist drops below this level
 132  * only the superuser may continue to allocate blocks. This may
 133  * be set to 0 if no reserve of free blocks is deemed necessary,
 134  * however throughput drops by fifty percent if the file system
 135  * is run at between 90% and 100% full; thus the default value of
 136  * fs_minfree is 10%. With 10% free space, fragmentation is not a
 137  * problem, so we choose to optimize for time.
 138  */
 139 #define MINFREE         10
 140 #define DEFAULTOPT      FS_OPTTIME
 141 
 142 /*
 143  * ROTDELAY gives the minimum number of milliseconds to initiate
 144  * another disk transfer on the same cylinder. It is no longer used
 145  * and will always default to 0.
 146  */
 147 #define ROTDELAY        0
 148 
 149 /*
 150  * MAXBLKPG determines the maximum number of data blocks which are
 151  * placed in a single cylinder group. The default is one indirect
 152  * block worth of data blocks.
 153  */
 154 #define MAXBLKPG(bsize) ((bsize) / sizeof (daddr32_t))
 155 
 156 /*
 157  * Each file system has a number of inodes statically allocated.
 158  * We allocate one inode slot per NBPI bytes, expecting this
 159  * to be far more than we will ever need.
 160  */
 161 #define NBPI            2048    /* Number Bytes Per Inode */
 162 #define MTB_NBPI        (MB)    /* Number Bytes Per Inode for multi-terabyte */
 163 
 164 /*
 165  * Disks are assumed to rotate at 60HZ, unless otherwise specified.
 166  */
 167 #define DEFHZ           60
 168 
 169 /*
 170  * Cylinder group related limits.
 171  *
 172  * For each cylinder we keep track of the availability of blocks at different
 173  * rotational positions, so that we can lay out the data to be picked
 174  * up with minimum rotational latency.  NRPOS is the number of rotational
 175  * positions which we distinguish.  With NRPOS 8 the resolution of our
 176  * summary information is 2ms for a typical 3600 rpm drive.
 177  */
 178 #define NRPOS           8       /* number distinct rotational positions */
 179 
 180 #ifdef DEBUG
 181 #define dprintf(x)      printf x
 182 #else
 183 #define dprintf(x)
 184 #endif
 185 
 186 /*
 187  * For the -N option, when calculating the backup superblocks, do not print
 188  * them if we are not really sure. We may have to try an alternate method of
 189  * arriving at the superblocks. So defer printing till a handful of superblocks
 190  * look good.
 191  */
 192 #define tprintf(x)      if (Nflag && retry) \
 193                                 (void) strncat(tmpbuf, x, strlen(x)); \
 194                         else \
 195                                 (void) fprintf(stderr, x);
 196 
 197 #define ALTSB           32      /* Location of first backup superblock */
 198 
 199 /*
 200  * range_check "user_supplied" flag values.
 201  */
 202 #define RC_DEFAULT      0
 203 #define RC_KEYWORD      1
 204 #define RC_POSITIONAL   2
 205 
 206 /*
 207  * ufs hole
 208  */
 209 #define UFS_HOLE        -1
 210 
 211 #ifndef STANDALONE
 212 #include        <stdio.h>
 213 #include        <sys/mnttab.h>
 214 #endif
 215 
 216 #include        <stdlib.h>
 217 #include        <unistd.h>
 218 #include        <malloc.h>
 219 #include        <string.h>
 220 #include        <strings.h>
 221 #include        <ctype.h>
 222 #include        <errno.h>
 223 #include        <sys/param.h>
 224 #include        <time.h>
 225 #include        <sys/types.h>
 226 #include        <sys/sysmacros.h>
 227 #include        <sys/vnode.h>
 228 #include        <sys/fs/ufs_fsdir.h>
 229 #include        <sys/fs/ufs_inode.h>
 230 #include        <sys/fs/ufs_fs.h>
 231 #include        <sys/fs/ufs_log.h>
 232 #include        <sys/mntent.h>
 233 #include        <sys/filio.h>
 234 #include        <limits.h>
 235 #include        <sys/int_const.h>
 236 #include        <signal.h>
 237 #include        <sys/efi_partition.h>

 238 #include        "roll_log.h"
 239 
 240 #define bcopy(f, t, n)    (void) memcpy(t, f, n)
 241 #define bzero(s, n)     (void) memset(s, 0, n)
 242 #define bcmp(s, d, n)   memcmp(s, d, n)
 243 
 244 #define index(s, r)     strchr(s, r)
 245 #define rindex(s, r)    strrchr(s, r)
 246 
 247 #include        <sys/stat.h>
 248 #include        <sys/statvfs.h>
 249 #include        <locale.h>
 250 #include        <fcntl.h>
 251 #include        <sys/isa_defs.h>  /* for ENDIAN defines */
 252 #include        <sys/vtoc.h>
 253 
 254 #include        <sys/dkio.h>
 255 #include        <sys/asynch.h>
 256 
 257 extern offset_t llseek();
 258 extern char     *getfullblkname();
 259 extern long     lrand48();
 260 
 261 extern int      optind;
 262 extern char     *optarg;
 263 
 264 
 265 /*
 266  * The size of a cylinder group is calculated by CGSIZE. The maximum size
 267  * is limited by the fact that cylinder groups are at most one block.
 268  * Its size is derived from the size of the maps maintained in the
 269  * cylinder group and the (struct cg) size.
 270  */
 271 #define CGSIZE(fs) \
 272         /* base cg              */ (sizeof (struct cg) + \
 273         /* blktot size  */ (fs)->fs_cpg * sizeof (long) + \
 274         /* blks size    */ (fs)->fs_cpg * (fs)->fs_nrpos * sizeof (short) + \
 275         /* inode map    */ howmany((fs)->fs_ipg, NBBY) + \
 276         /* block map */ howmany((fs)->fs_cpg * (fs)->fs_spc / NSPF(fs), NBBY))
 277 
 278 /*
 279  * We limit the size of the inode map to be no more than a
 280  * third of the cylinder group space, since we must leave at
 281  * least an equal amount of space for the block map.
 282  *
 283  * N.B.: MAXIpG must be a multiple of INOPB(fs).
 284  */
 285 #define MAXIpG(fs)      roundup((fs)->fs_bsize * NBBY / 3, INOPB(fs))
 286 
 287 /*
 288  * Same as MAXIpG, but parameterized by the block size (b) and the
 289  * cylinder group divisor (d), which is the reciprocal of the fraction of the
 290  * cylinder group overhead block that is used for the inode map.  So for
 291  * example, if d = 5, the macro's computation assumes that 1/5 of the
 292  * cylinder group overhead block can be dedicated to the inode map.
 293  */
 294 #define MAXIpG_B(b, d)  roundup((b) * NBBY / (d), (b) / sizeof (struct dinode))
 295 
 296 #define UMASK           0755
 297 #define MAXINOPB        (MAXBSIZE / sizeof (struct dinode))
 298 #define POWEROF2(num)   (((num) & ((num) - 1)) == 0)
 299 #define MB              (1024*1024)
 300 #define BETWEEN(x, l, h)        ((x) >= (l) && (x) <= (h))
 301 
 302 /*
 303  * Used to set the inode generation number. Since both inodes and dinodes
 304  * are dealt with, we really need a pointer to an icommon here.
 305  */
 306 #define IRANDOMIZE(icp) (icp)->ic_gen = lrand48();
 307 
 308 /*
 309  * Flags for number()
 310  */
 311 #define ALLOW_PERCENT   0x01    /* allow trailing `%' on number */
 312 #define ALLOW_MS1       0x02    /* allow trailing `ms', state 1 */
 313 #define ALLOW_MS2       0x04    /* allow trailing `ms', state 2 */
 314 #define ALLOW_END_ONLY  0x08    /* must be at end of number & suffixes */
 315 
 316 #define MAXAIO  1000    /* maximum number of outstanding I/O's we'll manage */
 317 #define BLOCK   1       /* block in aiowait */
 318 #define NOBLOCK 0       /* don't block in aiowait */
 319 
 320 #define RELEASE 1       /* free an aio buffer after use */
 321 #define SAVE    0       /* don't free the buffer */
 322 
 323 typedef struct aio_trans {
 324         aio_result_t resultbuf;
 325         diskaddr_t bno;
 326         char *buffer;
 327         int size;
 328         int release;
 329         struct aio_trans *next;
 330 } aio_trans;
 331 
 332 typedef struct aio_results {
 333         int max;
 334         int outstanding;
 335         int maxpend;
 336         aio_trans *trans;
 337 } aio_results;
 338 
 339 int aio_inited = 0;
 340 aio_results results;
 341 
 342 /*
 343  * Allow up to MAXBUF aio requests that each have a unique buffer.
 344  * More aio's might be done, but not using memory through the getbuf()
 345  * interface.  This can be raised, but you run into the potential of
 346  * using more memory than is physically available on the machine,
 347  * and if you start swapping, you can forget about performance.
 348  * To prevent this, we also limit the total memory used for a given
 349  * type of buffer to MAXBUFMEM.
 350  *
 351  * Tests indicate a cylinder group's worth of inodes takes:
 352  *
 353  *      NBPI    Size of Inode Buffer
 354  *       2k     1688k
 355  *       8k      424k
 356  *
 357  * initcg() stores all the inodes for a cylinder group in one buffer,
 358  * so allowing 20 buffers could take 32 MB if not limited by MAXBUFMEM.
 359  */
 360 #define MAXBUF          20
 361 #define MAXBUFMEM       (8 * 1024 * 1024)
 362 
 363 /*
 364  * header information for buffers managed by getbuf() and freebuf()
 365  */
 366 typedef struct bufhdr {
 367         struct bufhdr *head;
 368         struct bufhdr *next;
 369 } bufhdr;
 370 
 371 int bufhdrsize;
 372 
 373 bufhdr inodebuf = { NULL, NULL };
 374 bufhdr cgsumbuf = { NULL, NULL };
 375 
 376 #define SECTORS_PER_TERABYTE    (1LL << 31)
 377 /*
 378  * The following constant specifies an upper limit for file system size
 379  * that is actually a lot bigger than we expect to support with UFS. (Since
 380  * it's specified in sectors, the file system size would be 2**44 * 512,
 381  * which is 2**53, which is 8192 Terabytes.)  However, it's useful
 382  * for checking the basic sanity of a size value that is input on the
 383  * command line.
 384  */
 385 #define FS_SIZE_UPPER_LIMIT     0x100000000000LL
 386 
 387 /*
 388  * Forward declarations
 389  */
 390 static char *getbuf(bufhdr *bufhead, int size);
 391 static void freebuf(char *buf);
 392 static void freetrans(aio_trans *transp);
 393 static aio_trans *get_aiop();
 394 static aio_trans *wait_for_write(int block);
 395 static void initcg(int cylno);
 396 static void fsinit();
 397 static int makedir(struct direct *protodir, int entries);
 398 static void iput(struct inode *ip);
 399 static void rdfs(diskaddr_t bno, int size, char *bf);
 400 static void wtfs(diskaddr_t bno, int size, char *bf);
 401 static void awtfs(diskaddr_t bno, int size, char *bf, int release);
 402 static void wtfs_breakup(diskaddr_t bno, int size, char *bf);
 403 static int isblock(struct fs *fs, unsigned char *cp, int h);
 404 static void clrblock(struct fs *fs, unsigned char *cp, int h);
 405 static void setblock(struct fs *fs, unsigned char *cp, int h);
 406 static void usage(void) __NORETURN;
 407 static void dump_fscmd(char *fsys, int fsi);
 408 static uint64_t number(uint64_t d_value, char *param, int flags);
 409 static int match(char *s);
 410 static char checkopt(char *optim);
 411 static char checkmtb(char *mtbarg);
 412 static void range_check(long *varp, char *name, long minimum,
 413     long maximum, long def_val, int user_supplied);
 414 static void range_check_64(uint64_t *varp, char *name, uint64_t minimum,
 415     uint64_t maximum, uint64_t def_val, int user_supplied);
 416 static daddr32_t alloc(int size, int mode);
 417 static diskaddr_t get_max_size(int fd);
 418 static long get_max_track_size(int fd);
 419 static void block_sigint(sigset_t *old_mask);
 420 static void unblock_sigint(sigset_t *old_mask);
 421 static void recover_from_sigint(int signum);
 422 static int confirm_abort(void);
 423 static int getaline(FILE *fp, char *loc, int maxlen);
 424 static void flush_writes(void);
 425 static long compute_maxcpg(long, long, long, long, long);
 426 static int in_64bit_mode(void);
 427 static int validate_size(int fd, diskaddr_t size);
 428 static void dump_sblock(void);
 429 
 430 /*
 431  * Workaround for mkfs to function properly on disks attached to XMIT 2.X
 432  * controller. If the address is not aligned at 8 byte boundary, mkfs on
 433  * disks attached to XMIT 2.X controller exhibts un-predictable behaviour.
 434  */
 435 #define XMIT_2_X_ALIGN  8
 436 #pragma align XMIT_2_X_ALIGN(fsun, altfsun, cgun)
 437 
 438 union {
 439         struct fs fs;
 440         char pad[SBSIZE];
 441 } fsun, altfsun;
 442 #define sblock  fsun.fs
 443 #define altsblock       altfsun.fs
 444 
 445 struct  csum *fscs;
 446 
 447 union cgun {
 448         struct cg cg;
 449         char pad[MAXBSIZE];
 450 } cgun;
 451 
 452 #define acg     cgun.cg
 453 /*
 454  * Size of screen in cols in which to fit output
 455  */
 456 #define WIDTH   80
 457 
 458 struct dinode zino[MAXBSIZE / sizeof (struct dinode)];
 459 
 460 /*
 461  * file descriptors used for rdfs(fsi) and wtfs(fso).
 462  * Initialized to an illegal file descriptor number.
 463  */
 464 int     fsi = -1;
 465 int     fso = -1;
 466 
 467 /*
 468  * The BIG parameter is machine dependent.  It should be a longlong integer
 469  * constant that can be used by the number parser to check the validity
 470  * of numeric parameters.
 471  */
 472 
 473 #define BIG             0x7fffffffffffffffLL
 474 
 475 /* Used to indicate to number() that a bogus value should cause us to exit */
 476 #define NO_DEFAULT      LONG_MIN
 477 
 478 /*
 479  * INVALIDSBLIMIT is the number of bad backup superblocks that will be
 480  * tolerated before we decide to try arriving at a different set of them
 481  * using a different logic. This is applicable for non-EFI disks only.
 482  */
 483 #define INVALIDSBLIMIT  10
 484 
 485 /*
 486  * The *_flag variables are used to indicate that the user specified
 487  * the values, rather than that we made them up ourselves.  We can
 488  * complain about the user giving us bogus values.
 489  */
 490 
 491 /* semi-constants */
 492 long    sectorsize = DEV_BSIZE;         /* bytes/sector from param.h */
 493 long    bbsize = BBSIZE;                /* boot block size */
 494 long    sbsize = SBSIZE;                /* superblock size */
 495 
 496 /* parameters */
 497 diskaddr_t      fssize_db;              /* file system size in disk blocks */
 498 diskaddr_t      fssize_frag;            /* file system size in frags */
 499 long    cpg;                            /* cylinders/cylinder group */
 500 int     cpg_flag = RC_DEFAULT;
 501 long    rotdelay = -1;                  /* rotational delay between blocks */
 502 int     rotdelay_flag = RC_DEFAULT;
 503 long    maxcontig;                      /* max contiguous blocks to allocate */
 504 int     maxcontig_flag = RC_DEFAULT;
 505 long    nsect = DFLNSECT;               /* sectors per track */
 506 int     nsect_flag = RC_DEFAULT;
 507 long    ntrack = DFLNTRAK;              /* tracks per cylinder group */
 508 int     ntrack_flag = RC_DEFAULT;
 509 long    bsize = DESBLKSIZE;             /* filesystem block size */
 510 int     bsize_flag = RC_DEFAULT;
 511 long    fragsize = DESFRAGSIZE;         /* filesystem fragment size */
 512 int     fragsize_flag = RC_DEFAULT;
 513 long    minfree = MINFREE;              /* fs_minfree */
 514 int     minfree_flag = RC_DEFAULT;
 515 long    rps = DEFHZ;                    /* revolutions/second of drive */
 516 int     rps_flag = RC_DEFAULT;
 517 long    nbpi = NBPI;                    /* number of bytes per inode */
 518 int     nbpi_flag = RC_DEFAULT;
 519 long    nrpos = NRPOS;                  /* number of rotational positions */
 520 int     nrpos_flag = RC_DEFAULT;
 521 long    apc = 0;                        /* alternate sectors per cylinder */
 522 int     apc_flag = RC_DEFAULT;
 523 char    opt = 't';                      /* optimization style, `t' or `s' */
 524 char    mtb = 'n';                      /* multi-terabyte format, 'y' or 'n' */
 525 #define DEFAULT_SECT_TRAK_CPG   (nsect_flag == RC_DEFAULT && \
 526                                 ntrack_flag == RC_DEFAULT && \
 527                                 cpg_flag == RC_DEFAULT)
 528 
 529 long    debug = 0;                      /* enable debugging output */
 530 
 531 int     spc_flag = 0;                   /* alternate sectors specified or */
 532                                         /* found */
 533 
 534 /* global state */
 535 int     Nflag;          /* do not write to disk */
 536 int     mflag;          /* return the command line used to create this FS */
 537 int     rflag;          /* report the superblock in an easily-parsed form */
 538 int     Rflag;          /* dump the superblock in binary */
 539 char    *fsys;
 540 time_t  mkfstime;
 541 char    *string;
 542 int     label_type;
 543 
 544 /*
 545  * logging support
 546  */
 547 int     islog;                  /* true if ufs logging is enabled */
 548 int     islogok;                /* true if ufs log state is good */
 549 int     waslog;                 /* true when ufs logging disabled during grow */
 550 
 551 /*
 552  * growfs defines, globals, and forward references
 553  */
 554 #define NOTENOUGHSPACE 33
 555 int             grow;
 556 #define GROW_WITH_DEFAULT_TRAK  (grow && ntrack_flag == RC_DEFAULT)
 557 
 558 static int      Pflag;          /* probe to which size the fs can be grown */
 559 int             ismounted;
 560 char            *directory;
 561 diskaddr_t      grow_fssize;
 562 long            grow_fs_size;
 563 long            grow_fs_ncg;
 564 diskaddr_t              grow_fs_csaddr;
 565 long            grow_fs_cssize;
 566 int             grow_fs_clean;
 567 struct csum     *grow_fscs;
 568 diskaddr_t              grow_sifrag;
 569 int             test;
 570 int             testforce;
 571 diskaddr_t              testfrags;
 572 int             inlockexit;
 573 int             isbad;
 574 
 575 void            lockexit(int) __NORETURN;
 576 void            randomgeneration(void);
 577 void            checksummarysize(void);
 578 int             checksblock(struct fs, int);
 579 void            growinit(char *);
 580 void            checkdev(char *, char  *);
 581 void            checkmount(struct mnttab *, char *);
 582 struct dinode   *gdinode(ino_t);
 583 int             csfraginrange(daddr32_t);
 584 struct csfrag   *findcsfrag(daddr32_t, struct csfrag **);
 585 void            checkindirect(ino_t, daddr32_t *, daddr32_t, int);
 586 void            addcsfrag(ino_t, daddr32_t, struct csfrag **);
 587 void            delcsfrag(daddr32_t, struct csfrag **);
 588 void            checkdirect(ino_t, daddr32_t *, daddr32_t *, int);
 589 void            findcsfragino(void);
 590 void            fixindirect(daddr32_t, int);
 591 void            fixdirect(caddr_t, daddr32_t, daddr32_t *, int);
 592 void            fixcsfragino(void);
 593 void            extendsummaryinfo(void);
 594 int             notenoughspace(void);
 595 void            unalloccsfragino(void);
 596 void            unalloccsfragfree(void);
 597 void            findcsfragfree(void);
 598 void            copycsfragino(void);
 599 void            rdcg(long);
 600 void            wtcg(void);
 601 void            flcg(void);
 602 void            allocfrags(long, daddr32_t *, long *);
 603 void            alloccsfragino(void);
 604 void            alloccsfragfree(void);
 605 void            freefrags(daddr32_t, long, long);
 606 int             findfreerange(long *, long *);
 607 void            resetallocinfo(void);
 608 void            extendcg(long);
 609 void            ulockfs(void);
 610 void            wlockfs(void);
 611 void            clockfs(void);
 612 void            wtsb(void);
 613 static int64_t  checkfragallocated(daddr32_t);
 614 static struct csum      *read_summaryinfo(struct fs *);
 615 static diskaddr_t       probe_summaryinfo();
 616 
 617 int
 618 main(int argc, char *argv[])
 619 {
 620         long i, mincpc, mincpg, ibpcl;
 621         long cylno, rpos, blk, j, warn = 0;
 622         long mincpgcnt, maxcpg;
 623         uint64_t used, bpcg, inospercg;
 624         long mapcramped, inodecramped;
 625         long postblsize, rotblsize, totalsbsize;
 626         FILE *mnttab;
 627         struct mnttab mntp;
 628         char *special;
 629         struct statvfs64 fs;
 630         struct dk_geom dkg;
 631         struct dk_minfo dkminfo;
 632         char pbuf[sizeof (uint64_t) * 3 + 1];
 633         char *tmpbuf;
 634         int width, plen;
 635         uint64_t num;
 636         int c, saverr;
 637         diskaddr_t max_fssize;
 638         long tmpmaxcontig = -1;
 639         struct sigaction sigact;
 640         uint64_t nbytes64;
 641         int remaining_cg;
 642         int do_dot = 0;
 643         int use_efi_dflts = 0, retry = 0, isremovable = 0, ishotpluggable = 0;
 644         int invalid_sb_cnt, ret, skip_this_sb, cg_too_small;
 645         int geom_nsect, geom_ntrack, geom_cpg;
 646 
 647         (void) setlocale(LC_ALL, "");
 648 
 649 #if !defined(TEXT_DOMAIN)
 650 #define TEXT_DOMAIN "SYS_TEST"
 651 #endif
 652         (void) textdomain(TEXT_DOMAIN);
 653 
 654         while ((c = getopt(argc, argv, "F:bmo:VPGM:T:t:")) != EOF) {
 655                 switch (c) {
 656 
 657                 case 'F':
 658                         string = optarg;
 659                         if (strcmp(string, "ufs") != 0)
 660                                 usage();
 661                         break;
 662 
 663                 case 'm':       /* return command line used to create this FS */
 664                         mflag++;
 665                         break;
 666 
 667                 case 'o':
 668                         /*
 669                          * ufs specific options.
 670                          */
 671                         string = optarg;
 672                         while (*string != '\0') {
 673                                 if (match("nsect=")) {
 674                                         nsect = number(DFLNSECT, "nsect", 0);
 675                                         nsect_flag = RC_KEYWORD;
 676                                 } else if (match("ntrack=")) {
 677                                         ntrack = number(DFLNTRAK, "ntrack", 0);
 678                                         ntrack_flag = RC_KEYWORD;
 679                                 } else if (match("bsize=")) {
 680                                         bsize = number(DESBLKSIZE, "bsize", 0);
 681                                         bsize_flag = RC_KEYWORD;
 682                                 } else if (match("fragsize=")) {
 683                                         fragsize = number(DESFRAGSIZE,
 684                                             "fragsize", 0);
 685                                         fragsize_flag = RC_KEYWORD;
 686                                 } else if (match("cgsize=")) {
 687                                         cpg = number(DESCPG, "cgsize", 0);
 688                                         cpg_flag = RC_KEYWORD;
 689                                 } else if (match("free=")) {
 690                                         minfree = number(MINFREE, "free",
 691                                             ALLOW_PERCENT);
 692                                         minfree_flag = RC_KEYWORD;
 693                                 } else if (match("maxcontig=")) {
 694                                         tmpmaxcontig =
 695                                             number(-1, "maxcontig", 0);
 696                                         maxcontig_flag = RC_KEYWORD;
 697                                 } else if (match("nrpos=")) {
 698                                         nrpos = number(NRPOS, "nrpos", 0);
 699                                         nrpos_flag = RC_KEYWORD;
 700                                 } else if (match("rps=")) {
 701                                         rps = number(DEFHZ, "rps", 0);
 702                                         rps_flag = RC_KEYWORD;
 703                                 } else if (match("nbpi=")) {
 704                                         nbpi = number(NBPI, "nbpi", 0);
 705                                         nbpi_flag = RC_KEYWORD;
 706                                 } else if (match("opt=")) {
 707                                         opt = checkopt(string);
 708                                 } else if (match("mtb=")) {
 709                                         mtb = checkmtb(string);
 710                                 } else if (match("apc=")) {
 711                                         apc = number(0, "apc", 0);
 712                                         apc_flag = RC_KEYWORD;
 713                                 } else if (match("gap=")) {
 714                                         (void) number(0, "gap", ALLOW_MS1);
 715                                         rotdelay = ROTDELAY;
 716                                         rotdelay_flag = RC_DEFAULT;
 717                                 } else if (match("debug=")) {
 718                                         debug = number(0, "debug", 0);
 719                                 } else if (match("N")) {
 720                                         Nflag++;
 721                                 } else if (match("calcsb")) {
 722                                         rflag++;
 723                                         Nflag++;
 724                                 } else if (match("calcbinsb")) {
 725                                         rflag++;
 726                                         Rflag++;
 727                                         Nflag++;
 728                                 } else if (*string == '\0') {
 729                                         break;
 730                                 } else {
 731                                         (void) fprintf(stderr, gettext(
 732                                             "illegal option: %s\n"), string);
 733                                         usage();
 734                                 }
 735 
 736                                 if (*string == ',') string++;
 737                                 if (*string == ' ') string++;
 738                         }
 739                         break;
 740 
 741                 case 'V':
 742                         {
 743                                 char    *opt_text;
 744                                 int     opt_count;
 745 
 746                                 (void) fprintf(stdout, gettext("mkfs -F ufs "));
 747                                 for (opt_count = 1; opt_count < argc;
 748                                     opt_count++) {
 749                                         opt_text = argv[opt_count];
 750                                         if (opt_text)
 751                                                 (void) fprintf(stdout, " %s ",
 752                                                     opt_text);
 753                                 }
 754                                 (void) fprintf(stdout, "\n");
 755                         }
 756                         break;
 757 
 758                 case 'b':       /* do nothing for this */
 759                         break;
 760 
 761                 case 'M':       /* grow the mounted file system */
 762                         directory = optarg;
 763 
 764                         /* FALLTHROUGH */
 765                 case 'G':       /* grow the file system */
 766                         grow = 1;
 767                         break;
 768                 case 'P':       /* probe the file system growing size   */
 769                         Pflag = 1;
 770                         grow = 1; /* probe mode implies fs growing      */
 771                         break;
 772                 case 'T':       /* For testing */
 773                         testforce = 1;
 774 
 775                         /* FALLTHROUGH */
 776                 case 't':
 777                         test = 1;
 778                         string = optarg;
 779                         testfrags = number(NO_DEFAULT, "testfrags", 0);
 780                         break;
 781 
 782                 case '?':
 783                         usage();
 784                         break;
 785                 }
 786         }
 787 #ifdef MKFS_DEBUG
 788         /*
 789          * Turning on MKFS_DEBUG causes mkfs to produce a filesystem
 790          * that can be reproduced by setting the time to 0 and seeding
 791          * the random number generator to a constant.
 792          */
 793         mkfstime = 0;   /* reproducible results */
 794 #else
 795         (void) time(&mkfstime);
 796 #endif
 797 
 798         if (optind >= (argc - 1)) {
 799                 if (optind > (argc - 1)) {
 800                         (void) fprintf(stderr,
 801                             gettext("special not specified\n"));
 802                         usage();
 803                 } else if (mflag == 0) {
 804                         (void) fprintf(stderr,
 805                             gettext("size not specified\n"));
 806                         usage();
 807                 }
 808         }
 809         argc -= optind;
 810         argv = &argv[optind];
 811 
 812         fsys = argv[0];
 813         fsi = open64(fsys, O_RDONLY);
 814         if (fsi < 0) {
 815                 (void) fprintf(stderr, gettext("%s: cannot open\n"), fsys);
 816                 lockexit(32);
 817         }
 818 
 819         if (mflag) {
 820                 dump_fscmd(fsys, fsi);
 821                 lockexit(0);
 822         }
 823 
 824         /*
 825          * The task of setting all of the configuration parameters for a
 826          * UFS file system is basically a matter of solving n equations
 827          * in m variables.  Typically, m is greater than n, so there is
 828          * usually more than one valid solution.  Since this is usually
 829          * an under-constrained problem, it's not always obvious what the
 830          * "best" configuration is.
 831          *
 832          * In general, the approach is to
 833          * 1. Determine the values for the file system parameters
 834          *    that are externally contrained and therefore not adjustable
 835          *    by mkfs (such as the device's size and maxtransfer size).
 836          * 2. Acquire the user's requested setting for all configuration
 837          *    values that can be set on the command line.
 838          * 3. Determine the final value of all configuration values, by
 839          *    the following approach:
 840          *      - set the file system block size (fs_bsize).  Although
 841          *        this could be regarded as an adjustable parameter, in
 842          *        fact, it's pretty much a constant.  At this time, it's
 843          *        generally set to 8k (with older hardware, it can
 844          *        sometimes make sense to set it to 4k, but those
 845          *        situations are pretty rare now).
 846          *      - re-adjust the maximum file system size based on the
 847          *        value of the file system block size.  Since the
 848          *        frag size can't be any larger than a file system
 849          *        block, and the number of frags in the file system
 850          *        has to fit into 31 bits, the file system block size
 851          *        affects the maximum file system size.
 852          *      - now that the real maximum file system is known, set the
 853          *        actual size of the file system to be created to
 854          *        MIN(requested size, maximum file system size).
 855          *      - now validate, and if necessary, adjust the following
 856          *        values:
 857          *              rotdelay
 858          *              nsect
 859          *              maxcontig
 860          *              apc
 861          *              frag_size
 862          *              rps
 863          *              minfree
 864          *              nrpos
 865          *              nrack
 866          *              nbpi
 867          *      - calculate maxcpg (the maximum value of the cylinders-per-
 868          *        cylinder-group configuration parameters).  There are two
 869          *        algorithms for calculating maxcpg:  an old one, which is
 870          *        used for file systems of less than 1 terabyte, and a
 871          *        new one, implemented in the function compute_maxcpg(),
 872          *        which is used for file systems of greater than 1 TB.
 873          *        The difference between them is that compute_maxcpg()
 874          *        really tries to maximize the cpg value.  The old
 875          *        algorithm fails to take advantage of smaller frags and
 876          *        lower inode density when determining the maximum cpg,
 877          *        and thus comes up with much lower numbers in some
 878          *        configurations.  At some point, we might use the
 879          *        new algorithm for determining maxcpg for all file
 880          *        systems, but at this time, the changes implemented for
 881          *        multi-terabyte UFS are NOT being automatically applied
 882          *        to UFS file systems of less than a terabyte (in the
 883          *        interest of not changing existing UFS policy too much
 884          *        until the ramifications of the changes are well-understood
 885          *        and have been evaluated for their effects on performance.)
 886          *      - check the current values of the configuration parameters
 887          *        against the various constraints imposed by UFS.  These
 888          *        include:
 889          *              * There must be at least one inode in each
 890          *                cylinder group.
 891          *              * The cylinder group overhead block, which
 892          *                contains the inode and frag bigmaps, must fit
 893          *                within one file system block.
 894          *              * The space required for inode maps should
 895          *                occupy no more than a third of the cylinder
 896          *                group overhead block.
 897          *              * The rotational position tables have to fit
 898          *                within the available space in the super block.
 899          *        Adjust the configuration values that can be adjusted
 900          *        so that these constraints are satisfied.  The
 901          *        configuration values that are adjustable are:
 902          *              * frag size
 903          *              * cylinders per group
 904          *              * inode density (can be increased)
 905          *              * number of rotational positions (the rotational
 906          *                position tables are eliminated altogether if
 907          *                there isn't enough room for them.)
 908          * 4. Set the values for all the dependent configuration
 909          *    values (those that aren't settable on the command
 910          *    line and which are completely dependent on the
 911          *    adjustable parameters).  This include cpc (cycles
 912          *    per cylinder, spc (sectors-per-cylinder), and many others.
 913          */
 914 
 915         /*
 916          * Figure out the partition size and initialize the label_type.
 917          */
 918         max_fssize = get_max_size(fsi);
 919 
 920         /*
 921          * Get and check positional arguments, if any.
 922          */
 923         switch (argc - 1) {
 924         default:
 925                 usage();
 926                 /*NOTREACHED*/
 927         case 15:
 928                 mtb = checkmtb(argv[15]);
 929                 /* FALLTHROUGH */
 930         case 14:
 931                 string = argv[14];
 932                 tmpmaxcontig = number(-1, "maxcontig", 0);
 933                 maxcontig_flag = RC_POSITIONAL;
 934                 /* FALLTHROUGH */
 935         case 13:
 936                 string = argv[13];
 937                 nrpos = number(NRPOS, "nrpos", 0);
 938                 nrpos_flag = RC_POSITIONAL;
 939                 /* FALLTHROUGH */
 940         case 12:
 941                 string = argv[12];
 942                 rotdelay = ROTDELAY;
 943                 rotdelay_flag = RC_DEFAULT;
 944                 /* FALLTHROUGH */
 945         case 11:
 946                 string = argv[11];
 947                 apc = number(0, "apc", 0);
 948                 apc_flag = RC_POSITIONAL;
 949                 /* FALLTHROUGH */
 950         case 10:
 951                 opt = checkopt(argv[10]);
 952                 /* FALLTHROUGH */
 953         case 9:
 954                 string = argv[9];
 955                 nbpi = number(NBPI, "nbpi", 0);
 956                 nbpi_flag = RC_POSITIONAL;
 957                 /* FALLTHROUGH */
 958         case 8:
 959                 string = argv[8];
 960                 rps = number(DEFHZ, "rps", 0);
 961                 rps_flag = RC_POSITIONAL;
 962                 /* FALLTHROUGH */
 963         case 7:
 964                 string = argv[7];
 965                 minfree = number(MINFREE, "free", ALLOW_PERCENT);
 966                 minfree_flag = RC_POSITIONAL;
 967                 /* FALLTHROUGH */
 968         case 6:
 969                 string = argv[6];
 970                 cpg = number(DESCPG, "cgsize", 0);
 971                 cpg_flag = RC_POSITIONAL;
 972                 /* FALLTHROUGH */
 973         case 5:
 974                 string = argv[5];
 975                 fragsize = number(DESFRAGSIZE, "fragsize", 0);
 976                 fragsize_flag = RC_POSITIONAL;
 977                 /* FALLTHROUGH */
 978         case 4:
 979                 string = argv[4];
 980                 bsize = number(DESBLKSIZE, "bsize", 0);
 981                 bsize_flag = RC_POSITIONAL;
 982                 /* FALLTHROUGH */
 983         case 3:
 984                 string = argv[3];
 985                 ntrack = number(DFLNTRAK, "ntrack", 0);
 986                 ntrack_flag = RC_POSITIONAL;
 987                 /* FALLTHROUGH */
 988         case 2:
 989                 string = argv[2];
 990                 nsect = number(DFLNSECT, "nsect", 0);
 991                 nsect_flag = RC_POSITIONAL;
 992                 /* FALLTHROUGH */
 993         case 1:
 994                 string = argv[1];
 995                 fssize_db = number(max_fssize, "size", 0);
 996         }
 997 
 998         /*
 999          * Initialize the parameters in the same way as newfs so that
1000          * newfs and mkfs would result in the same file system layout
1001          * for EFI labelled disks. Do this only in the absence of user
1002          * specified values for these parameters.
1003          */
1004         if (label_type == LABEL_TYPE_EFI) {
1005                 if (apc_flag == RC_DEFAULT) apc = 0;
1006                 if (nrpos_flag == RC_DEFAULT) nrpos = 1;
1007                 if (ntrack_flag == RC_DEFAULT) ntrack = DEF_TRACKS_EFI;
1008                 if (rps_flag == RC_DEFAULT) rps = DEFHZ;
1009                 if (nsect_flag == RC_DEFAULT) nsect = DEF_SECTORS_EFI;
1010         }
1011 
1012         if ((maxcontig_flag == RC_DEFAULT) || (tmpmaxcontig == -1) ||
1013             (maxcontig == -1)) {
1014                 long maxtrax = get_max_track_size(fsi);
1015                 maxcontig = maxtrax / bsize;
1016 
1017         } else {
1018                 maxcontig = tmpmaxcontig;
1019         }
1020         dprintf(("DeBuG maxcontig : %ld\n", maxcontig));
1021 
1022         if (rotdelay == -1) {   /* default by newfs and mkfs */
1023                 rotdelay = ROTDELAY;
1024         }
1025 
1026         if (cpg_flag == RC_DEFAULT) { /* If not explicity set, use default */
1027                 cpg = DESCPG;
1028         }
1029         dprintf(("DeBuG cpg : %ld\n", cpg));
1030 
1031         /*
1032          * Now that we have the semi-sane args, either positional, via -o,
1033          * or by defaulting, handle inter-dependencies and range checks.
1034          */
1035 
1036         /*
1037          * Settle the file system block size first, since it's a fixed
1038          * parameter once set and so many other parameters, including
1039          * max_fssize, depend on it.
1040          */
1041         range_check(&bsize, "bsize", MINBSIZE, MAXBSIZE, DESBLKSIZE,
1042             bsize_flag);
1043 
1044         if (!POWEROF2(bsize)) {
1045                 (void) fprintf(stderr,
1046                     gettext("block size must be a power of 2, not %ld\n"),
1047                     bsize);
1048                 bsize = DESBLKSIZE;
1049                 (void) fprintf(stderr,
1050                     gettext("mkfs: bsize reset to default %ld\n"),
1051                     bsize);
1052         }
1053 
1054         if (fssize_db > max_fssize && validate_size(fsi, fssize_db)) {
1055                 (void) fprintf(stderr, gettext(
1056                     "Warning: the requested size of this file system\n"
1057                     "(%lld sectors) is greater than the size of the\n"
1058                     "device reported by the driver (%lld sectors).\n"
1059                     "However, a read of the device at the requested size\n"
1060                     "does succeed, so the requested size will be used.\n"),
1061                     fssize_db, max_fssize);
1062                 max_fssize = fssize_db;
1063         }
1064         /*
1065          * Since the maximum allocatable unit (the frag) must be less than
1066          * or equal to bsize, and the number of frags must be less than or
1067          * equal to INT_MAX, the total size of the file system (in
1068          * bytes) must be less than or equal to bsize * INT_MAX.
1069          */
1070 
1071         if (max_fssize > ((diskaddr_t)bsize/DEV_BSIZE) * INT_MAX)
1072                 max_fssize = ((diskaddr_t)bsize/DEV_BSIZE) * INT_MAX;
1073 
1074         range_check_64(&fssize_db, "size", 1024LL, max_fssize, max_fssize, 1);
1075 
1076         if (fssize_db >= SECTORS_PER_TERABYTE) {
1077                 mtb = 'y';
1078                 if (!in_64bit_mode()) {
1079                         (void) fprintf(stderr, gettext(
1080 "mkfs:  Warning: Creating a file system greater than 1 terabyte on a\n"
1081 "       system running a 32-bit kernel.  This file system will not be\n"
1082 "       accessible until the system is rebooted with a 64-bit kernel.\n"));
1083                 }
1084         }
1085         dprintf(("DeBuG mtb : %c\n", mtb));
1086 
1087         /*
1088          * With newer and much larger disks, the newfs(1M) and mkfs_ufs(1M)
1089          * commands had problems in correctly handling the "native" geometries
1090          * for various storage devices.
1091          *
1092          * To handle the new age disks, mkfs_ufs(1M) will use the EFI style
1093          * for non-EFI disks that are larger than the CHS addressing limit
1094          * ( > 8GB approx ) and ignore the disk geometry information for
1095          * these drives. This is what is currently done for multi-terrabyte
1096          * filesystems on EFI disks.
1097          *
1098          * However if the user asked for a specific layout by supplying values
1099          * for even one of the three parameters (nsect, ntrack, cpg), honour
1100          * the user supplied parameters.
1101          *
1102          * Choosing EFI style or native geometry style can make a lot of
1103          * difference, because the size of a cylinder group is dependent on
1104          * this choice. This in turn means that the position of alternate
1105          * superblocks varies depending on the style chosen. It is not
1106          * necessary that all disks of size > CHSLIMIT have EFI style layout.
1107          * There can be disks which are > CHSLIMIT size, but have native
1108          * geometry style layout, thereby warranting the need for alternate
1109          * logic in superblock detection.
1110          */
1111         if (mtb != 'y' && (ntrack == -1 || GROW_WITH_DEFAULT_TRAK ||
1112             DEFAULT_SECT_TRAK_CPG)) {
1113                 /*
1114                  * "-1" indicates that we were called from newfs and ntracks
1115                  * was not specified in newfs command line. Calculate nsect
1116                  * and ntrack in the same manner as newfs.
1117                  *
1118                  * This is required because, the defaults for nsect and ntrack
1119                  * is hardcoded in mkfs, whereas to generate the alternate
1120                  * superblock locations for the -N option, there is a need for
1121                  * the geometry based values that newfs would have arrived at.
1122                  * Newfs would have arrived at these values as below.
1123                  */
1124                 if (label_type == LABEL_TYPE_EFI ||
1125                     label_type == LABEL_TYPE_OTHER) {
1126                         use_efi_dflts = 1;
1127                         retry = 1;
1128                 } else if (ioctl(fsi, DKIOCGGEOM, &dkg)) {
1129                         dprintf(("%s: Unable to read Disk geometry", fsys));
1130                         perror(gettext("Unable to read Disk geometry"));
1131                         lockexit(32);
1132                 } else {
1133                         nsect = dkg.dkg_nsect;
1134                         ntrack = dkg.dkg_nhead;
1135 #ifdef i386     /* Bug 1170182 */
1136                         if (ntrack > 32 && (ntrack % 16) != 0) {
1137                                 ntrack -= (ntrack % 16);
1138                         }
1139 #endif
1140                         if (ioctl(fsi, DKIOCREMOVABLE, &isremovable)) {
1141                                 dprintf(("DeBuG Unable to determine if %s is"
1142                                     " Removable Media. Proceeding with system"
1143                                     " determined parameters.\n", fsys));
1144                                 isremovable = 0;
1145                         }
1146                         if (ioctl(fsi, DKIOCHOTPLUGGABLE, &ishotpluggable)) {
1147                                 dprintf(("DeBuG Unable to determine if %s is"
1148                                     " Hotpluggable Media. Proceeding with "
1149                                     "system determined parameters.\n", fsys));
1150                                 ishotpluggable = 0;
1151                         }
1152                         if ((((diskaddr_t)dkg.dkg_ncyl * dkg.dkg_nhead *
1153                             dkg.dkg_nsect) > CHSLIMIT) || isremovable ||
1154                             ishotpluggable) {
1155                                 use_efi_dflts = 1;
1156                                 retry = 1;
1157                         }
1158                 }
1159         }
1160         dprintf(("DeBuG CHSLIMIT = %d geom = %llu\n", CHSLIMIT,
1161             (diskaddr_t)dkg.dkg_ncyl * dkg.dkg_nhead * dkg.dkg_nsect));
1162         dprintf(("DeBuG label_type = %d isremovable = %d ishotpluggable = %d "
1163             "use_efi_dflts = %d\n", label_type, isremovable, ishotpluggable,
1164             use_efi_dflts));
1165 
1166         /*
1167          * For the newfs -N case, even if the disksize is > CHSLIMIT, do not
1168          * blindly follow EFI style. If the fs_version indicates a geometry
1169          * based layout, try that one first. If it fails we can always try the
1170          * other logic.
1171          *
1172          * If we were called from growfs, we will have a problem if we mix
1173          * and match the filesystem creation and growth styles. For example,
1174          * if we create using EFI style, we have to also grow using EFI
1175          * style. So follow the style indicated by the fs_version.
1176          *
1177          * Read and verify the primary superblock. If it looks sane, use the
1178          * fs_version from the superblock. If the primary superblock does
1179          * not look good, read and verify the first alternate superblock at
1180          * ALTSB. Use the fs_version to decide whether to use the
1181          * EFI style logic or the old geometry based logic to calculate
1182          * the alternate superblock locations.
1183          */
1184         if ((Nflag && use_efi_dflts) || (grow)) {
1185                 if (grow && ntrack_flag != RC_DEFAULT)
1186                         goto start_fs_creation;
1187                 rdfs((diskaddr_t)(SBOFF / sectorsize), (int)sbsize,
1188                     (char *)&altsblock);
1189                 ret = checksblock(altsblock, 1);
1190 
1191                 if (!ret) {
1192                         if (altsblock.fs_magic == MTB_UFS_MAGIC) {
1193                                 mtb = 'y';
1194                                 goto start_fs_creation;
1195                         }
1196                         use_efi_dflts = (altsblock.fs_version ==
1197                             UFS_EFISTYLE4NONEFI_VERSION_2) ? 1 : 0;
1198                 } else {
1199                         /*
1200                          * The primary superblock didn't help in determining
1201                          * the fs_version. Try the first alternate superblock.
1202                          */
1203                         dprintf(("DeBuG checksblock() failed - error : %d"
1204                             " for sb : %d\n", ret, SBOFF/sectorsize));
1205                         rdfs((diskaddr_t)ALTSB, (int)sbsize,
1206                             (char *)&altsblock);
1207                         ret = checksblock(altsblock, 1);
1208 
1209                         if (!ret) {
1210                                 if (altsblock.fs_magic == MTB_UFS_MAGIC) {
1211                                         mtb = 'y';
1212                                         goto start_fs_creation;
1213                                 }
1214                                 use_efi_dflts = (altsblock.fs_version ==
1215                                     UFS_EFISTYLE4NONEFI_VERSION_2) ? 1 : 0;
1216                         }
1217                         dprintf(("DeBuG checksblock() returned : %d"
1218                             " for sb : %d\n", ret, ALTSB));
1219                 }
1220         }
1221 
1222         geom_nsect = nsect;
1223         geom_ntrack = ntrack;
1224         geom_cpg = cpg;
1225         dprintf(("DeBuG geom_nsect=%d, geom_ntrack=%d, geom_cpg=%d\n",
1226             geom_nsect, geom_ntrack, geom_cpg));
1227 
1228 start_fs_creation:
1229 retry_alternate_logic:
1230         invalid_sb_cnt = 0;
1231         cg_too_small = 0;
1232         if (use_efi_dflts) {
1233                 nsect = DEF_SECTORS_EFI;
1234                 ntrack = DEF_TRACKS_EFI;
1235                 cpg = DESCPG;
1236                 dprintf(("\nDeBuG Using EFI defaults\n"));
1237         } else {
1238                 nsect = geom_nsect;
1239                 ntrack = geom_ntrack;
1240                 cpg = geom_cpg;
1241                 dprintf(("\nDeBuG Using Geometry\n"));
1242                 /*
1243                  * 32K based on max block size of 64K, and rotational layout
1244                  * test of nsect <= (256 * sectors/block).  Current block size
1245                  * limit is not 64K, but it's growing soon.
1246                  */
1247                 range_check(&nsect, "nsect", 1, 32768, DFLNSECT, nsect_flag);
1248                 /*
1249                  * ntrack is the number of tracks per cylinder.
1250                  * The ntrack value must be between 1 and the total number of
1251                  * sectors in the file system.
1252                  */
1253                 range_check(&ntrack, "ntrack", 1,
1254                     fssize_db > INT_MAX ? INT_MAX : (uint32_t)fssize_db,
1255                     DFLNTRAK, ntrack_flag);
1256         }
1257 
1258         range_check(&apc, "apc", 0, nsect - 1, 0, apc_flag);
1259 
1260         if (mtb == 'y')
1261                 fragsize = bsize;
1262 
1263         range_check(&fragsize, "fragsize", sectorsize, bsize,
1264             MAX(bsize / MAXFRAG, MIN(DESFRAGSIZE, bsize)), fragsize_flag);
1265 
1266         if ((bsize / MAXFRAG) > fragsize) {
1267                 (void) fprintf(stderr, gettext(
1268 "fragment size %ld is too small, minimum with block size %ld is %ld\n"),
1269                     fragsize, bsize, bsize / MAXFRAG);
1270                 (void) fprintf(stderr,
1271                     gettext("mkfs: fragsize reset to minimum %ld\n"),
1272                     bsize / MAXFRAG);
1273                 fragsize = bsize / MAXFRAG;
1274         }
1275 
1276         if (!POWEROF2(fragsize)) {
1277                 (void) fprintf(stderr,
1278                     gettext("fragment size must be a power of 2, not %ld\n"),
1279                     fragsize);
1280                 fragsize = MAX(bsize / MAXFRAG, MIN(DESFRAGSIZE, bsize));
1281                 (void) fprintf(stderr,
1282                     gettext("mkfs: fragsize reset to %ld\n"),
1283                     fragsize);
1284         }
1285 
1286         /* At this point, bsize must be >= fragsize, so no need to check it */
1287 
1288         if (bsize < PAGESIZE) {
1289                 (void) fprintf(stderr, gettext(
1290                     "WARNING: filesystem block size (%ld) is smaller than "
1291                     "memory page size (%ld).\nResulting filesystem can not be "
1292                     "mounted on this system.\n\n"),
1293                     bsize, (long)PAGESIZE);
1294         }
1295 
1296         range_check(&rps, "rps", 1, 1000, DEFHZ, rps_flag);
1297         range_check(&minfree, "free", 0, 99, MINFREE, minfree_flag);
1298         range_check(&nrpos, "nrpos", 1, nsect, MIN(nsect, NRPOS), nrpos_flag);
1299 
1300         /*
1301          * nbpi is variable, but 2MB seems a reasonable upper limit,
1302          * as 4MB tends to cause problems (using otherwise-default
1303          * parameters).  The true limit is where we end up with one
1304          * inode per cylinder group.  If this file system is being
1305          * configured for multi-terabyte access, nbpi must be at least 1MB.
1306          */
1307         if (mtb == 'y' && nbpi < MTB_NBPI) {
1308                 if (nbpi_flag != RC_DEFAULT)
1309                         (void) fprintf(stderr, gettext("mkfs: bad value for "
1310                             "nbpi: must be at least 1048576 for multi-terabyte,"
1311                             " nbpi reset to default 1048576\n"));
1312                 nbpi = MTB_NBPI;
1313         }
1314 
1315         if (mtb == 'y')
1316                 range_check(&nbpi, "nbpi", MTB_NBPI, 2 * MB, MTB_NBPI,
1317                     nbpi_flag);
1318         else
1319                 range_check(&nbpi, "nbpi", DEV_BSIZE, 2 * MB, NBPI, nbpi_flag);
1320 
1321         /*
1322          * maxcpg is another variably-limited parameter.  Calculate
1323          * the limit based on what we've got for its dependent
1324          * variables.  Effectively, it's how much space is left in the
1325          * superblock after all the other bits are accounted for.  We
1326          * only fill in sblock fields so we can use MAXIpG.
1327          *
1328          * If the calculation of maxcpg below (for the mtb == 'n'
1329          * case) is changed, update newfs as well.
1330          *
1331          * For old-style, non-MTB format file systems, use the old
1332          * algorithm for calculating the maximum cylinder group size,
1333          * even though it limits the cylinder group more than necessary.
1334          * Since layout can affect performance, we don't want to change
1335          * the default layout for non-MTB file systems at this time.
1336          * However, for MTB file systems, use the new maxcpg calculation,
1337          * which really maxes out the cylinder group size.
1338          */
1339 
1340         sblock.fs_bsize = bsize;
1341         sblock.fs_inopb = sblock.fs_bsize / sizeof (struct dinode);
1342 
1343         if (mtb == 'n') {
1344                 maxcpg = (bsize - sizeof (struct cg) -
1345                     howmany(MAXIpG(&sblock), NBBY)) /
1346                     (sizeof (long) + nrpos * sizeof (short) +
1347                     nsect / (MAXFRAG * NBBY));
1348         } else {
1349                 maxcpg = compute_maxcpg(bsize, fragsize, nbpi, nrpos,
1350                     nsect * ntrack);
1351         }
1352 
1353         dprintf(("DeBuG cpg : %ld\n", cpg));
1354         /*
1355          * Increase the cpg to maxcpg if either newfs was invoked
1356          * with -T option or if mkfs wants to create a mtb file system
1357          * and if the user has not specified the cpg.
1358          */
1359         if (cpg == -1 || (mtb == 'y' && cpg_flag == RC_DEFAULT))
1360                 cpg = maxcpg;
1361         dprintf(("DeBuG cpg : %ld\n", cpg));
1362 
1363         /*
1364          * mincpg is variable in complex ways, so we really can't
1365          * do a sane lower-end limit check at this point.
1366          */
1367         range_check(&cpg, "cgsize", 1, maxcpg, MIN(maxcpg, DESCPG), cpg_flag);
1368 
1369         /*
1370          * get the controller info
1371          */
1372         islog = 0;
1373         islogok = 0;
1374         waslog = 0;
1375 
1376         /*
1377          * Do not grow the file system, but print on stdout the maximum
1378          * size in sectors to which the file system can be increased.
1379          * The calculated size is limited by fssize_db.
1380          * Note that we don't lock the filesystem and therefore under rare
1381          * conditions (the filesystem is mounted, the free block count is
1382          * almost zero, and the superuser is still changing it) the calculated
1383          * size can be imprecise.
1384          */
1385         if (Pflag) {
1386                 (void) printf("%llu\n", probe_summaryinfo());
1387                 exit(0);
1388         }
1389 
1390         /*
1391          * If we're growing an existing filesystem, then we're about
1392          * to start doing things that can require recovery efforts if
1393          * we get interrupted, so make sure we get a chance to do so.
1394          */
1395         if (grow) {
1396                 sigact.sa_handler = recover_from_sigint;
1397                 sigemptyset(&sigact.sa_mask);
1398                 sigact.sa_flags = SA_RESTART;
1399 
1400                 if (sigaction(SIGINT, &sigact, (struct sigaction *)NULL) < 0) {
1401                         perror(gettext("Could not register SIGINT handler"));
1402                         lockexit(3);
1403                 }
1404         }
1405 
1406         if (!Nflag) {
1407                 /*
1408                  * Check if MNTTAB is trustable
1409                  */
1410                 if (statvfs64(MNTTAB, &fs) < 0) {
1411                         (void) fprintf(stderr, gettext("can't statvfs %s\n"),
1412                             MNTTAB);
1413                         exit(32);
1414                 }
1415 
1416                 if (strcmp(MNTTYPE_MNTFS, fs.f_basetype) != 0) {
1417                         (void) fprintf(stderr, gettext(
1418                             "%s file system type is not %s, can't mkfs\n"),
1419                             MNTTAB, MNTTYPE_MNTFS);
1420                         exit(32);
1421                 }
1422 
1423                 special = getfullblkname(fsys);
1424                 checkdev(fsys, special);
1425 
1426                 /*
1427                  * If we found the block device name,
1428                  * then check the mount table.
1429                  * if mounted, and growing write lock the file system
1430                  *
1431                  */
1432                 if ((special != NULL) && (*special != '\0')) {
1433                         if ((mnttab = fopen(MNTTAB, "r")) == NULL) {
1434                                 (void) fprintf(stderr, gettext(
1435                                     "can't open %s\n"), MNTTAB);
1436                                 exit(32);
1437                         }
1438                         while ((getmntent(mnttab, &mntp)) == NULL) {
1439                                 if (grow) {
1440                                         checkmount(&mntp, special);
1441                                         continue;
1442                                 }
1443                                 if (strcmp(special, mntp.mnt_special) == 0) {
1444                                         (void) fprintf(stderr, gettext(
1445                                             "%s is mounted, can't mkfs\n"),
1446                                             special);
1447                                         exit(32);
1448                                 }
1449                         }
1450                         (void) fclose(mnttab);
1451                 }
1452 
1453                 if (directory && (ismounted == 0)) {
1454                         (void) fprintf(stderr, gettext("%s is not mounted\n"),
1455                             special);
1456                         lockexit(32);
1457                 }
1458 
1459                 fso = (grow) ? open64(fsys, O_WRONLY) : creat64(fsys, 0666);
1460                 if (fso < 0) {
1461                         saverr = errno;
1462                         (void) fprintf(stderr,
1463                             gettext("%s: cannot create: %s\n"),
1464                             fsys, strerror(saverr));
1465                         lockexit(32);
1466                 }
1467 
1468         } else {
1469 
1470                 /*
1471                  * For the -N case, a file descriptor is needed for the llseek()
1472                  * in wtfs(). See the comment in wtfs() for more information.
1473                  *
1474                  * Get a file descriptor that's read-only so that this code
1475                  * doesn't accidentally write to the file.
1476                  */
1477                 fso = open64(fsys, O_RDONLY);
1478                 if (fso < 0) {
1479                         saverr = errno;
1480                         (void) fprintf(stderr, gettext("%s: cannot open: %s\n"),
1481                             fsys, strerror(saverr));
1482                         lockexit(32);
1483                 }
1484         }
1485 
1486         /*
1487          * Check the media sector size
1488          */
1489         if (ioctl(fso, DKIOCGMEDIAINFO, &dkminfo) != -1) {
1490                 if (dkminfo.dki_lbsize != 0 &&
1491                     POWEROF2(dkminfo.dki_lbsize / DEV_BSIZE) &&
1492                     dkminfo.dki_lbsize != DEV_BSIZE) {
1493                         fprintf(stderr,
1494                             gettext("The device sector size %u is not "
1495                             "supported by ufs!\n"), dkminfo.dki_lbsize);
1496                         (void) close(fso);
1497                         exit(1);
1498                 }
1499         }
1500 
1501         /*
1502          * seed random # generator (for ic_generation)
1503          */
1504 #ifdef MKFS_DEBUG
1505         srand48(12962); /* reproducible results */
1506 #else
1507         srand48((long)(time((time_t *)NULL) + getpid()));
1508 #endif
1509 
1510         if (grow) {
1511                 growinit(fsys);
1512                 goto grow00;
1513         }
1514 
1515         /*
1516          * Validate the given file system size.
1517          * Verify that its last block can actually be accessed.
1518          *
1519          * Note: it's ok to use sblock as a buffer because it is immediately
1520          * overwritten by the rdfs() of the superblock in the next line.
1521          *
1522          * ToDo: Because the size checking is done in rdfs()/wtfs(), the
1523          * error message for specifying an illegal size is very unfriendly.
1524          * In the future, one could replace the rdfs()/wtfs() calls
1525          * below with in-line calls to read() or write(). This allows better
1526          * error messages to be put in place.
1527          */
1528         rdfs(fssize_db - 1, (int)sectorsize, (char *)&sblock);
1529 
1530         /*
1531          * make the fs unmountable
1532          */
1533         rdfs((diskaddr_t)(SBOFF / sectorsize), (int)sbsize, (char *)&sblock);
1534         sblock.fs_magic = -1;
1535         sblock.fs_clean = FSBAD;
1536         sblock.fs_state = FSOKAY - sblock.fs_time;
1537         wtfs((diskaddr_t)(SBOFF / sectorsize), (int)sbsize, (char *)&sblock);
1538         bzero(&sblock, (size_t)sbsize);
1539 
1540         sblock.fs_nsect = nsect;
1541         sblock.fs_ntrak = ntrack;
1542 
1543         /*
1544          * Validate specified/determined spc
1545          * and calculate minimum cylinders per group.
1546          */
1547 
1548         /*
1549          * sectors/cyl = tracks/cyl * sectors/track
1550          */
1551         sblock.fs_spc = sblock.fs_ntrak * sblock.fs_nsect;
1552 
1553 grow00:
1554         if (apc_flag) {
1555                 sblock.fs_spc -= apc;
1556         }
1557         /*
1558          * Have to test for this separately from apc_flag, due to
1559          * the growfs case....
1560          */
1561         if (sblock.fs_spc != sblock.fs_ntrak * sblock.fs_nsect) {
1562                 spc_flag = 1;
1563         }
1564         if (grow)
1565                 goto grow10;
1566 
1567         sblock.fs_nrpos = nrpos;
1568         sblock.fs_bsize = bsize;
1569         sblock.fs_fsize = fragsize;
1570         sblock.fs_minfree = minfree;
1571 
1572 grow10:
1573         if (nbpi < sblock.fs_fsize) {
1574                 (void) fprintf(stderr, gettext(
1575                 "warning: wasteful data byte allocation / inode (nbpi):\n"));
1576                 (void) fprintf(stderr, gettext(
1577                     "%ld smaller than allocatable fragment size of %d\n"),
1578                     nbpi, sblock.fs_fsize);
1579         }
1580         if (grow)
1581                 goto grow20;
1582 
1583         if (opt == 's')
1584                 sblock.fs_optim = FS_OPTSPACE;
1585         else
1586                 sblock.fs_optim = FS_OPTTIME;
1587 
1588         sblock.fs_bmask = ~(sblock.fs_bsize - 1);
1589         sblock.fs_fmask = ~(sblock.fs_fsize - 1);
1590         /*
1591          * Planning now for future expansion.
1592          */
1593 #if defined(_BIG_ENDIAN)
1594                 sblock.fs_qbmask.val[0] = 0;
1595                 sblock.fs_qbmask.val[1] = ~sblock.fs_bmask;
1596                 sblock.fs_qfmask.val[0] = 0;
1597                 sblock.fs_qfmask.val[1] = ~sblock.fs_fmask;
1598 #endif
1599 #if defined(_LITTLE_ENDIAN)
1600                 sblock.fs_qbmask.val[0] = ~sblock.fs_bmask;
1601                 sblock.fs_qbmask.val[1] = 0;
1602                 sblock.fs_qfmask.val[0] = ~sblock.fs_fmask;
1603                 sblock.fs_qfmask.val[1] = 0;
1604 #endif
1605         for (sblock.fs_bshift = 0, i = sblock.fs_bsize; i > 1; i >>= 1)
1606                 sblock.fs_bshift++;
1607         for (sblock.fs_fshift = 0, i = sblock.fs_fsize; i > 1; i >>= 1)
1608                 sblock.fs_fshift++;
1609         sblock.fs_frag = numfrags(&sblock, sblock.fs_bsize);
1610         for (sblock.fs_fragshift = 0, i = sblock.fs_frag; i > 1; i >>= 1)
1611                 sblock.fs_fragshift++;
1612         if (sblock.fs_frag > MAXFRAG) {
1613                 (void) fprintf(stderr, gettext(
1614         "fragment size %d is too small, minimum with block size %d is %d\n"),
1615                     sblock.fs_fsize, sblock.fs_bsize,
1616                     sblock.fs_bsize / MAXFRAG);
1617                 lockexit(32);
1618         }
1619         sblock.fs_nindir = sblock.fs_bsize / sizeof (daddr32_t);
1620         sblock.fs_inopb = sblock.fs_bsize / sizeof (struct dinode);
1621         sblock.fs_nspf = sblock.fs_fsize / sectorsize;
1622         for (sblock.fs_fsbtodb = 0, i = NSPF(&sblock); i > 1; i >>= 1)
1623                 sblock.fs_fsbtodb++;
1624 
1625         /*
1626          * Compute the super-block, cylinder group, and inode blocks.
1627          * Note that these "blkno" are really fragment addresses.
1628          * For example, on an 8K/1K (block/fragment) system, fs_sblkno is 16,
1629          * fs_cblkno is 24, and fs_iblkno is 32. This is why CGSIZE is so
1630          * important: only 1 FS block is allocated for the cg struct (fragment
1631          * numbers 24 through 31).
1632          */
1633         sblock.fs_sblkno =
1634             roundup(howmany(bbsize + sbsize, sblock.fs_fsize), sblock.fs_frag);
1635         sblock.fs_cblkno = (daddr32_t)(sblock.fs_sblkno +
1636             roundup(howmany(sbsize, sblock.fs_fsize), sblock.fs_frag));
1637         sblock.fs_iblkno = sblock.fs_cblkno + sblock.fs_frag;
1638 
1639         sblock.fs_cgoffset = roundup(
1640             howmany(sblock.fs_nsect, NSPF(&sblock)), sblock.fs_frag);
1641         for (sblock.fs_cgmask = -1, i = sblock.fs_ntrak; i > 1; i >>= 1)
1642                 sblock.fs_cgmask <<= 1;
1643         if (!POWEROF2(sblock.fs_ntrak))
1644                 sblock.fs_cgmask <<= 1;
1645         /*
1646          * Validate specified/determined spc
1647          * and calculate minimum cylinders per group.
1648          */
1649 
1650         for (sblock.fs_cpc = NSPB(&sblock), i = sblock.fs_spc;
1651             sblock.fs_cpc > 1 && (i & 1) == 0;
1652             sblock.fs_cpc >>= 1, i >>= 1)
1653                 /* void */;
1654         mincpc = sblock.fs_cpc;
1655 
1656         /* if these calculations are changed, check dump_fscmd also */
1657         bpcg = (uint64_t)sblock.fs_spc * sectorsize;
1658         inospercg = (uint64_t)roundup(bpcg / sizeof (struct dinode),
1659             INOPB(&sblock));
1660         if (inospercg > MAXIpG(&sblock))
1661                 inospercg = MAXIpG(&sblock);
1662         used = (uint64_t)(sblock.fs_iblkno + inospercg /
1663             INOPF(&sblock)) * NSPF(&sblock);
1664         mincpgcnt = (long)howmany((uint64_t)sblock.fs_cgoffset *
1665             (~sblock.fs_cgmask) + used, sblock.fs_spc);
1666         mincpg = roundup(mincpgcnt, mincpc);
1667         /*
1668          * Insure that cylinder group with mincpg has enough space
1669          * for block maps
1670          */
1671         sblock.fs_cpg = mincpg;
1672         sblock.fs_ipg = (int32_t)inospercg;
1673         mapcramped = 0;
1674 
1675         /*
1676          * Make sure the cg struct fits within the file system block.
1677          * Use larger block sizes until it fits
1678          */
1679         while (CGSIZE(&sblock) > sblock.fs_bsize) {
1680                 mapcramped = 1;
1681                 if (sblock.fs_bsize < MAXBSIZE) {
1682                         sblock.fs_bsize <<= 1;
1683                         if ((i & 1) == 0) {
1684                                 i >>= 1;
1685                         } else {
1686                                 sblock.fs_cpc <<= 1;
1687                                 mincpc <<= 1;
1688                                 mincpg = roundup(mincpgcnt, mincpc);
1689                                 sblock.fs_cpg = mincpg;
1690                         }
1691                         sblock.fs_frag <<= 1;
1692                         sblock.fs_fragshift += 1;
1693                         if (sblock.fs_frag <= MAXFRAG)
1694                                 continue;
1695                 }
1696 
1697                 /*
1698                  * Looped far enough. The fragment is now as large as the
1699                  * filesystem block!
1700                  */
1701                 if (sblock.fs_fsize == sblock.fs_bsize) {
1702                         (void) fprintf(stderr, gettext(
1703                     "There is no block size that can support this disk\n"));
1704                         lockexit(32);
1705                 }
1706 
1707                 /*
1708                  * Try a larger fragment. Double the fragment size.
1709                  */
1710                 sblock.fs_frag >>= 1;
1711                 sblock.fs_fragshift -= 1;
1712                 sblock.fs_fsize <<= 1;
1713                 sblock.fs_nspf <<= 1;
1714         }
1715         /*
1716          * Insure that cylinder group with mincpg has enough space for inodes
1717          */
1718         inodecramped = 0;
1719         used *= sectorsize;
1720         nbytes64 = (uint64_t)mincpg * bpcg - used;
1721         inospercg = (uint64_t)roundup((nbytes64 / nbpi), INOPB(&sblock));
1722         sblock.fs_ipg = (int32_t)inospercg;
1723         while (inospercg > MAXIpG(&sblock)) {
1724                 inodecramped = 1;
1725                 if (mincpc == 1 || sblock.fs_frag == 1 ||
1726                     sblock.fs_bsize == MINBSIZE)
1727                         break;
1728                 nbytes64 = (uint64_t)mincpg * bpcg - used;
1729                 (void) fprintf(stderr,
1730                     gettext("With a block size of %d %s %lu\n"),
1731                     sblock.fs_bsize, gettext("minimum bytes per inode is"),
1732                     (uint32_t)(nbytes64 / MAXIpG(&sblock) + 1));
1733                 sblock.fs_bsize >>= 1;
1734                 sblock.fs_frag >>= 1;
1735                 sblock.fs_fragshift -= 1;
1736                 mincpc >>= 1;
1737                 sblock.fs_cpg = roundup(mincpgcnt, mincpc);
1738                 if (CGSIZE(&sblock) > sblock.fs_bsize) {
1739                         sblock.fs_bsize <<= 1;
1740                         break;
1741                 }
1742                 mincpg = sblock.fs_cpg;
1743                 nbytes64 = (uint64_t)mincpg * bpcg - used;
1744                 inospercg = (uint64_t)roundup((nbytes64 / nbpi),
1745                     INOPB(&sblock));
1746                 sblock.fs_ipg = (int32_t)inospercg;
1747         }
1748         if (inodecramped) {
1749                 if (inospercg > MAXIpG(&sblock)) {
1750                         nbytes64 = (uint64_t)mincpg * bpcg - used;
1751                         (void) fprintf(stderr, gettext(
1752                             "Minimum bytes per inode is %d\n"),
1753                             (uint32_t)(nbytes64 / MAXIpG(&sblock) + 1));
1754                 } else if (!mapcramped) {
1755                         (void) fprintf(stderr, gettext(
1756             "With %ld bytes per inode, minimum cylinders per group is %ld\n"),
1757                             nbpi, mincpg);
1758                 }
1759         }
1760         if (mapcramped) {
1761                 (void) fprintf(stderr, gettext(
1762                     "With %d sectors per cylinder, minimum cylinders "
1763                     "per group is %ld\n"),
1764                     sblock.fs_spc, mincpg);
1765         }
1766         if (inodecramped || mapcramped) {
1767                 /*
1768                  * To make this at least somewhat comprehensible in
1769                  * the world of i18n, figure out what we're going to
1770                  * say and then say it all at one time.  The days of
1771                  * needing to scrimp on string space are behind us....
1772                  */
1773                 if ((sblock.fs_bsize != bsize) &&
1774                     (sblock.fs_fsize != fragsize)) {
1775                         (void) fprintf(stderr, gettext(
1776             "This requires the block size to be changed from %ld to %d\n"
1777             "and the fragment size to be changed from %ld to %d\n"),
1778                             bsize, sblock.fs_bsize,
1779                             fragsize, sblock.fs_fsize);
1780                 } else if (sblock.fs_bsize != bsize) {
1781                         (void) fprintf(stderr, gettext(
1782             "This requires the block size to be changed from %ld to %d\n"),
1783                             bsize, sblock.fs_bsize);
1784                 } else if (sblock.fs_fsize != fragsize) {
1785                         (void) fprintf(stderr, gettext(
1786             "This requires the fragment size to be changed from %ld to %d\n"),
1787                             fragsize, sblock.fs_fsize);
1788                 } else {
1789                         (void) fprintf(stderr, gettext(
1790             "Unable to make filesystem fit with the given constraints\n"));
1791                 }
1792                 (void) fprintf(stderr, gettext(
1793                     "Please re-run mkfs with corrected parameters\n"));
1794                 lockexit(32);
1795         }
1796         /*
1797          * Calculate the number of cylinders per group
1798          */
1799         sblock.fs_cpg = cpg;
1800         if (sblock.fs_cpg % mincpc != 0) {
1801                 (void) fprintf(stderr, gettext(
1802                     "Warning: cylinder groups must have a multiple "
1803                     "of %ld cylinders with the given\n         parameters\n"),
1804                     mincpc);
1805                 sblock.fs_cpg = roundup(sblock.fs_cpg, mincpc);
1806                 (void) fprintf(stderr, gettext("Rounded cgsize up to %d\n"),
1807                     sblock.fs_cpg);
1808         }
1809         /*
1810          * Must insure there is enough space for inodes
1811          */
1812         /* if these calculations are changed, check dump_fscmd also */
1813         nbytes64 = (uint64_t)sblock.fs_cpg * bpcg - used;
1814         sblock.fs_ipg = roundup((uint32_t)(nbytes64 / nbpi), INOPB(&sblock));
1815 
1816         /*
1817          * Slim down cylinders per group, until the inodes can fit.
1818          */
1819         while (sblock.fs_ipg > MAXIpG(&sblock)) {
1820                 inodecramped = 1;
1821                 sblock.fs_cpg -= mincpc;
1822                 nbytes64 = (uint64_t)sblock.fs_cpg * bpcg - used;
1823                 sblock.fs_ipg = roundup((uint32_t)(nbytes64 / nbpi),
1824                     INOPB(&sblock));
1825         }
1826         /*
1827          * Must insure there is enough space to hold block map.
1828          * Cut down on cylinders per group, until the cg struct fits in a
1829          * filesystem block.
1830          */
1831         while (CGSIZE(&sblock) > sblock.fs_bsize) {
1832                 mapcramped = 1;
1833                 sblock.fs_cpg -= mincpc;
1834                 nbytes64 = (uint64_t)sblock.fs_cpg * bpcg - used;
1835                 sblock.fs_ipg = roundup((uint32_t)(nbytes64 / nbpi),
1836                     INOPB(&sblock));
1837         }
1838         sblock.fs_fpg = (sblock.fs_cpg * sblock.fs_spc) / NSPF(&sblock);
1839         if ((sblock.fs_cpg * sblock.fs_spc) % NSPB(&sblock) != 0) {
1840                 (void) fprintf(stderr,
1841                 gettext("newfs: panic (fs_cpg * fs_spc) %% NSPF != 0\n"));
1842                 lockexit(32);
1843         }
1844         if (sblock.fs_cpg < mincpg) {
1845                 (void) fprintf(stderr, gettext(
1846 "With the given parameters, cgsize must be at least %ld; please re-run mkfs\n"),
1847                     mincpg);
1848                 lockexit(32);
1849         }
1850         sblock.fs_cgsize = fragroundup(&sblock, CGSIZE(&sblock));
1851 grow20:
1852         /*
1853          * Now have size for file system and nsect and ntrak.
1854          * Determine number of cylinders and blocks in the file system.
1855          */
1856         fssize_frag = (int64_t)dbtofsb(&sblock, fssize_db);
1857         if (fssize_frag > INT_MAX) {
1858                 (void) fprintf(stderr, gettext(
1859 "There are too many fragments in the system, increase fragment size\n"),
1860                     mincpg);
1861                 lockexit(32);
1862         }
1863         sblock.fs_size = (int32_t)fssize_frag;
1864         sblock.fs_ncyl = (int32_t)(fssize_frag * NSPF(&sblock) / sblock.fs_spc);
1865         if (fssize_frag * NSPF(&sblock) >
1866             (uint64_t)sblock.fs_ncyl * sblock.fs_spc) {
1867                 sblock.fs_ncyl++;
1868                 warn = 1;
1869         }
1870         if (sblock.fs_ncyl < 1) {
1871                 (void) fprintf(stderr, gettext(
1872                     "file systems must have at least one cylinder\n"));
1873                 lockexit(32);
1874         }
1875         if (grow)
1876                 goto grow30;
1877         /*
1878          * Determine feasability/values of rotational layout tables.
1879          *
1880          * The size of the rotational layout tables is limited by the size
1881          * of the file system block, fs_bsize.  The amount of space
1882          * available for tables is calculated as (fs_bsize - sizeof (struct
1883          * fs)).  The size of these tables is inversely proportional to the
1884          * block size of the file system. The size increases if sectors per
1885          * track are not powers of two, because more cylinders must be
1886          * described by the tables before the rotational pattern repeats
1887          * (fs_cpc).
1888          */
1889         sblock.fs_postblformat = FS_DYNAMICPOSTBLFMT;
1890         sblock.fs_sbsize = fragroundup(&sblock, sizeof (struct fs));
1891         sblock.fs_npsect = sblock.fs_nsect;
1892         if (sblock.fs_ntrak == 1) {
1893                 sblock.fs_cpc = 0;
1894                 goto next;
1895         }
1896         postblsize = sblock.fs_nrpos * sblock.fs_cpc * sizeof (short);
1897         rotblsize = sblock.fs_cpc * sblock.fs_spc / NSPB(&sblock);
1898         totalsbsize = sizeof (struct fs) + rotblsize;
1899 
1900         /* do static allocation if nrpos == 8 and fs_cpc == 16  */
1901         if (sblock.fs_nrpos == 8 && sblock.fs_cpc <= 16) {
1902                 /* use old static table space */
1903                 sblock.fs_postbloff = (char *)(&sblock.fs_opostbl[0][0]) -
1904                     (char *)(&sblock.fs_link);
1905                 sblock.fs_rotbloff = &sblock.fs_space[0] -
1906                     (uchar_t *)(&sblock.fs_link);
1907         } else {
1908                 /* use 4.3 dynamic table space */
1909                 sblock.fs_postbloff = &sblock.fs_space[0] -
1910                     (uchar_t *)(&sblock.fs_link);
1911                 sblock.fs_rotbloff = sblock.fs_postbloff + postblsize;
1912                 totalsbsize += postblsize;
1913         }
1914         if (totalsbsize > sblock.fs_bsize ||
1915             sblock.fs_nsect > (1 << NBBY) * NSPB(&sblock)) {
1916                 (void) fprintf(stderr, gettext(
1917                     "Warning: insufficient space in super block for\n"
1918                     "rotational layout tables with nsect %d, ntrack %d, "
1919                     "and nrpos %d.\nOmitting tables - file system "
1920                     "performance may be impaired.\n"),
1921                     sblock.fs_nsect, sblock.fs_ntrak, sblock.fs_nrpos);
1922 
1923                 /*
1924                  * Setting fs_cpc to 0 tells alloccgblk() in ufs_alloc.c to
1925                  * ignore the positional layout table and rotational
1926                  * position table.
1927                  */
1928                 sblock.fs_cpc = 0;
1929                 goto next;
1930         }
1931         sblock.fs_sbsize = fragroundup(&sblock, totalsbsize);
1932 
1933 
1934         /*
1935          * calculate the available blocks for each rotational position
1936          */
1937         for (cylno = 0; cylno < sblock.fs_cpc; cylno++)
1938                 for (rpos = 0; rpos < sblock.fs_nrpos; rpos++)
1939                         fs_postbl(&sblock, cylno)[rpos] = -1;
1940         for (i = (rotblsize - 1) * sblock.fs_frag;
1941             i >= 0; i -= sblock.fs_frag) {
1942                 cylno = cbtocylno(&sblock, i);
1943                 rpos = cbtorpos(&sblock, i);
1944                 blk = fragstoblks(&sblock, i);
1945                 if (fs_postbl(&sblock, cylno)[rpos] == -1)
1946                         fs_rotbl(&sblock)[blk] = 0;
1947                 else
1948                         fs_rotbl(&sblock)[blk] =
1949                             fs_postbl(&sblock, cylno)[rpos] - blk;
1950                 fs_postbl(&sblock, cylno)[rpos] = blk;
1951         }
1952 next:
1953 grow30:
1954         /*
1955          * Compute/validate number of cylinder groups.
1956          * Note that if an excessively large filesystem is specified
1957          * (e.g., more than 16384 cylinders for an 8K filesystem block), it
1958          * does not get detected until checksummarysize()
1959          */
1960         sblock.fs_ncg = sblock.fs_ncyl / sblock.fs_cpg;
1961         if (sblock.fs_ncyl % sblock.fs_cpg)
1962                 sblock.fs_ncg++;
1963         sblock.fs_dblkno = sblock.fs_iblkno + sblock.fs_ipg / INOPF(&sblock);
1964         i = MIN(~sblock.fs_cgmask, sblock.fs_ncg - 1);
1965         ibpcl = cgdmin(&sblock, i) - cgbase(&sblock, i);
1966         if (ibpcl >= sblock.fs_fpg) {
1967                 (void) fprintf(stderr, gettext(
1968                     "inode blocks/cyl group (%d) >= data blocks (%d)\n"),
1969                     cgdmin(&sblock, i) - cgbase(&sblock, i) / sblock.fs_frag,
1970                     sblock.fs_fpg / sblock.fs_frag);
1971                 if ((ibpcl < 0) || (sblock.fs_fpg < 0)) {
1972                         (void) fprintf(stderr, gettext(
1973             "number of cylinders per cylinder group (%d) must be decreased.\n"),
1974                             sblock.fs_cpg);
1975                 } else {
1976                         (void) fprintf(stderr, gettext(
1977             "number of cylinders per cylinder group (%d) must be increased.\n"),
1978                             sblock.fs_cpg);
1979                 }
1980                 (void) fprintf(stderr, gettext(
1981 "Note that cgsize may have been adjusted to allow struct cg to fit.\n"));
1982                 lockexit(32);
1983         }
1984         j = sblock.fs_ncg - 1;
1985         if ((i = fssize_frag - j * sblock.fs_fpg) < sblock.fs_fpg &&
1986             cgdmin(&sblock, j) - cgbase(&sblock, j) > i) {
1987                 (void) fprintf(stderr, gettext(
1988                     "Warning: inode blocks/cyl group (%d) >= data "
1989                     "blocks (%ld) in last\n    cylinder group. This "
1990                     "implies %ld sector(s) cannot be allocated.\n"),
1991                     (cgdmin(&sblock, j) - cgbase(&sblock, j)) / sblock.fs_frag,
1992                     i / sblock.fs_frag, i * NSPF(&sblock));
1993                 /*
1994                  * If there is only one cylinder group and that is not even
1995                  * big enough to hold the inodes, exit.
1996                  */
1997                 if (sblock.fs_ncg == 1)
1998                         cg_too_small = 1;
1999                 sblock.fs_ncg--;
2000                 sblock.fs_ncyl = sblock.fs_ncg * sblock.fs_cpg;
2001                 sblock.fs_size = fssize_frag =
2002                     (int64_t)sblock.fs_ncyl * (int64_t)sblock.fs_spc /
2003                     (int64_t)NSPF(&sblock);
2004                 warn = 0;
2005         }
2006         if (warn && !spc_flag) {
2007                 (void) fprintf(stderr, gettext(
2008                     "Warning: %d sector(s) in last cylinder unallocated\n"),
2009                     sblock.fs_spc - (uint32_t)(fssize_frag * NSPF(&sblock) -
2010                     (uint64_t)(sblock.fs_ncyl - 1) * sblock.fs_spc));
2011         }
2012         /*
2013          * fill in remaining fields of the super block
2014          */
2015 
2016         /*
2017          * The csum records are stored in cylinder group 0, starting at
2018          * cgdmin, the first data block.
2019          */
2020         sblock.fs_csaddr = cgdmin(&sblock, 0);
2021         sblock.fs_cssize =
2022             fragroundup(&sblock, sblock.fs_ncg * sizeof (struct csum));
2023         i = sblock.fs_bsize / sizeof (struct csum);
2024         sblock.fs_csmask = ~(i - 1);
2025         for (sblock.fs_csshift = 0; i > 1; i >>= 1)
2026                 sblock.fs_csshift++;
2027         fscs = (struct csum *)calloc(1, sblock.fs_cssize);
2028 
2029         checksummarysize();
2030         if (mtb == 'y') {
2031                 sblock.fs_magic = MTB_UFS_MAGIC;
2032                 sblock.fs_version = MTB_UFS_VERSION_1;
2033         } else {
2034                 sblock.fs_magic = FS_MAGIC;
2035                 if (use_efi_dflts)
2036                         sblock.fs_version = UFS_EFISTYLE4NONEFI_VERSION_2;
2037                 else
2038                         sblock.fs_version = UFS_VERSION_MIN;
2039         }
2040 
2041         if (grow) {
2042                 bcopy((caddr_t)grow_fscs, (caddr_t)fscs, (int)grow_fs_cssize);
2043                 extendsummaryinfo();
2044                 goto grow40;
2045         }
2046         sblock.fs_rotdelay = rotdelay;
2047         sblock.fs_maxcontig = maxcontig;
2048         sblock.fs_maxbpg = MAXBLKPG(sblock.fs_bsize);
2049 
2050         sblock.fs_rps = rps;
2051         sblock.fs_cgrotor = 0;
2052         sblock.fs_cstotal.cs_ndir = 0;
2053         sblock.fs_cstotal.cs_nbfree = 0;
2054         sblock.fs_cstotal.cs_nifree = 0;
2055         sblock.fs_cstotal.cs_nffree = 0;
2056         sblock.fs_fmod = 0;
2057         sblock.fs_ronly = 0;
2058         sblock.fs_time = mkfstime;
2059         sblock.fs_state = FSOKAY - sblock.fs_time;
2060         sblock.fs_clean = FSCLEAN;
2061 grow40:
2062 
2063         /*
2064          * If all that's needed is a dump of the superblock we
2065          * would use by default, we've got it now.  So, splat it
2066          * out and leave.
2067          */
2068         if (rflag) {
2069                 dump_sblock();
2070                 lockexit(0);
2071         }
2072         /*
2073          * Dump out summary information about file system.
2074          */
2075         (void) fprintf(stderr, gettext(
2076             "%s:\t%lld sectors in %d cylinders of %d tracks, %d sectors\n"),
2077             fsys, (uint64_t)sblock.fs_size * NSPF(&sblock), sblock.fs_ncyl,
2078             sblock.fs_ntrak, sblock.fs_nsect);
2079         (void) fprintf(stderr, gettext(
2080             "\t%.1fMB in %d cyl groups (%d c/g, %.2fMB/g, %d i/g)\n"),
2081             (float)sblock.fs_size * sblock.fs_fsize / MB, sblock.fs_ncg,
2082             sblock.fs_cpg, (float)sblock.fs_fpg * sblock.fs_fsize / MB,
2083             sblock.fs_ipg);
2084 
2085         tmpbuf = calloc(sblock.fs_ncg / 50 + 500, 1);
2086         if (tmpbuf == NULL) {
2087                 perror("calloc");
2088                 lockexit(32);
2089         }
2090         if (cg_too_small) {
2091                 (void) fprintf(stderr, gettext("File system creation failed. "
2092                     "There is only one cylinder group and\nthat is "
2093                     "not even big enough to hold the inodes.\n"));
2094                 lockexit(32);
2095         }
2096         /*
2097          * Now build the cylinders group blocks and
2098          * then print out indices of cylinder groups.
2099          */
2100         tprintf(gettext(
2101             "super-block backups (for fsck -F ufs -o b=#) at:\n"));
2102         for (width = cylno = 0; cylno < sblock.fs_ncg && cylno < 10; cylno++) {
2103                 if ((grow == 0) || (cylno >= grow_fs_ncg))
2104                         initcg(cylno);
2105                 num = fsbtodb(&sblock, (uint64_t)cgsblock(&sblock, cylno));
2106                 /*
2107                  * If Nflag and if the disk is larger than the CHSLIMIT,
2108                  * then sanity test the superblocks before reporting. If there
2109                  * are too many superblocks which look insane, we have
2110                  * to retry with alternate logic. If both methods have
2111                  * failed, then our efforts to arrive at alternate
2112                  * superblocks failed, so complain and exit.
2113                  */
2114                 if (Nflag && retry) {
2115                         skip_this_sb = 0;
2116                         rdfs((diskaddr_t)num, sbsize, (char *)&altsblock);
2117                         ret = checksblock(altsblock, 1);
2118                         if (ret) {
2119                                 skip_this_sb = 1;
2120                                 invalid_sb_cnt++;
2121                                 dprintf(("DeBuG checksblock() failed - error :"
2122                                     " %d for sb : %llu invalid_sb_cnt : %d\n",
2123                                     ret, num, invalid_sb_cnt));
2124                         } else {
2125                                 /*
2126                                  * Though the superblock looks sane, verify if
2127                                  * the fs_version in the superblock and the
2128                                  * logic that we are using to arrive at the
2129                                  * superblocks match.
2130                                  */
2131                                 if (use_efi_dflts && altsblock.fs_version
2132                                     != UFS_EFISTYLE4NONEFI_VERSION_2) {
2133                                         skip_this_sb = 1;
2134                                         invalid_sb_cnt++;
2135                                 }
2136                         }
2137                         if (invalid_sb_cnt >= INVALIDSBLIMIT) {
2138                                 if (retry > 1) {
2139                                         (void) fprintf(stderr, gettext(
2140                                             "Error determining alternate "
2141                                             "superblock locations\n"));
2142                                         free(tmpbuf);
2143                                         lockexit(32);
2144                                 }
2145                                 retry++;
2146                                 use_efi_dflts = !use_efi_dflts;
2147                                 free(tmpbuf);
2148                                 goto retry_alternate_logic;
2149                         }
2150                         if (skip_this_sb)
2151                                 continue;
2152                 }
2153                 (void) sprintf(pbuf, " %llu,", num);
2154                 plen = strlen(pbuf);
2155                 if ((width + plen) > (WIDTH - 1)) {
2156                         width = plen;
2157                         tprintf("\n");
2158                 } else {
2159                         width += plen;
2160                 }
2161                 if (Nflag && retry)
2162                         (void) strncat(tmpbuf, pbuf, strlen(pbuf));
2163                 else
2164                         (void) fprintf(stderr, "%s", pbuf);
2165         }
2166         tprintf("\n");
2167 
2168         remaining_cg = sblock.fs_ncg - cylno;
2169 
2170         /*
2171          * If there are more than 300 cylinder groups still to be
2172          * initialized, print a "." for every 50 cylinder groups.
2173          */
2174         if (remaining_cg > 300) {
2175                 tprintf(gettext("Initializing cylinder groups:\n"));
2176                 do_dot = 1;
2177         }
2178 
2179         /*
2180          * Now initialize all cylinder groups between the first ten
2181          * and the last ten.
2182          *
2183          * If the number of cylinder groups was less than 10, all of the
2184          * cylinder group offsets would have printed in the last loop
2185          * and cylno will already be equal to sblock.fs_ncg and so this
2186          * loop will not be entered.  If there are less than 20 cylinder
2187          * groups, cylno is already less than fs_ncg - 10, so this loop
2188          * won't be entered in that case either.
2189          */
2190 
2191         i = 0;
2192         for (; cylno < sblock.fs_ncg - 10; cylno++) {
2193                 if ((grow == 0) || (cylno >= grow_fs_ncg))
2194                         initcg(cylno);
2195                 if (do_dot && cylno % 50 == 0) {
2196                         tprintf(".");
2197                         i++;
2198                         if (i == WIDTH - 1) {
2199                                 tprintf("\n");
2200                                 i = 0;
2201                         }
2202                 }
2203         }
2204 
2205         /*
2206          * Now print the cylinder group offsets for the last 10
2207          * cylinder groups, if any are left.
2208          */
2209 
2210         if (do_dot) {
2211                 tprintf(gettext(
2212             "\nsuper-block backups for last 10 cylinder groups at:\n"));
2213         }
2214         for (width = 0; cylno < sblock.fs_ncg; cylno++) {
2215                 if ((grow == 0) || (cylno >= grow_fs_ncg))
2216                         initcg(cylno);
2217                 num = fsbtodb(&sblock, (uint64_t)cgsblock(&sblock, cylno));
2218                 if (Nflag && retry) {
2219                         skip_this_sb = 0;
2220                         rdfs((diskaddr_t)num, sbsize, (char *)&altsblock);
2221                         ret = checksblock(altsblock, 1);
2222                         if (ret) {
2223                                 skip_this_sb = 1;
2224                                 invalid_sb_cnt++;
2225                                 dprintf(("DeBuG checksblock() failed - error :"
2226                                     " %d for sb : %llu invalid_sb_cnt : %d\n",
2227                                     ret, num, invalid_sb_cnt));
2228                         } else {
2229                                 /*
2230                                  * Though the superblock looks sane, verify if
2231                                  * the fs_version in the superblock and the
2232                                  * logic that we are using to arrive at the
2233                                  * superblocks match.
2234                                  */
2235                                 if (use_efi_dflts && altsblock.fs_version
2236                                     != UFS_EFISTYLE4NONEFI_VERSION_2) {
2237                                         skip_this_sb = 1;
2238                                         invalid_sb_cnt++;
2239                                 }
2240                         }
2241                         if (invalid_sb_cnt >= INVALIDSBLIMIT) {
2242                                 if (retry > 1) {
2243                                         (void) fprintf(stderr, gettext(
2244                                             "Error determining alternate "
2245                                             "superblock locations\n"));
2246                                         free(tmpbuf);
2247                                         lockexit(32);
2248                                 }
2249                                 retry++;
2250                                 use_efi_dflts = !use_efi_dflts;
2251                                 free(tmpbuf);
2252                                 goto retry_alternate_logic;
2253                         }
2254                         if (skip_this_sb)
2255                                 continue;
2256                 }
2257                 /* Don't print ',' for the last superblock */
2258                 if (cylno == sblock.fs_ncg-1)
2259                         (void) sprintf(pbuf, " %llu", num);
2260                 else
2261                         (void) sprintf(pbuf, " %llu,", num);
2262                 plen = strlen(pbuf);
2263                 if ((width + plen) > (WIDTH - 1)) {
2264                         width = plen;
2265                         tprintf("\n");
2266                 } else {
2267                         width += plen;
2268                 }
2269                 if (Nflag && retry)
2270                         (void) strncat(tmpbuf, pbuf, strlen(pbuf));
2271                 else
2272                         (void) fprintf(stderr, "%s", pbuf);
2273         }
2274         tprintf("\n");
2275         if (Nflag) {
2276                 if (retry)
2277                         (void) fprintf(stderr, "%s", tmpbuf);
2278                 free(tmpbuf);
2279                 lockexit(0);
2280         }
2281 
2282         free(tmpbuf);
2283         if (grow)
2284                 goto grow50;
2285 
2286         /*
2287          * Now construct the initial file system,
2288          * then write out the super-block.
2289          */
2290         fsinit();
2291 grow50:
2292         /*
2293          * write the superblock and csum information
2294          */
2295         wtsb();
2296 
2297         /*
2298          * extend the last cylinder group in the original file system
2299          */
2300         if (grow) {
2301                 extendcg(grow_fs_ncg-1);
2302                 wtsb();
2303         }
2304 
2305         /*
2306          * Write out the duplicate super blocks to the first 10
2307          * cylinder groups (or fewer, if there are fewer than 10
2308          * cylinder groups).
2309          */
2310         for (cylno = 0; cylno < sblock.fs_ncg && cylno < 10; cylno++)
2311                 awtfs(fsbtodb(&sblock, (uint64_t)cgsblock(&sblock, cylno)),
2312                     (int)sbsize, (char *)&sblock, SAVE);
2313 
2314         /*
2315          * Now write out duplicate super blocks to the remaining
2316          * cylinder groups.  In the case of multi-terabyte file
2317          * systems, just write out the super block to the last ten
2318          * cylinder groups (or however many are left).
2319          */
2320         if (mtb == 'y') {
2321                 if (sblock.fs_ncg <= 10)
2322                         cylno = sblock.fs_ncg;
2323                 else if (sblock.fs_ncg <= 20)
2324                         cylno = 10;
2325                 else
2326                         cylno = sblock.fs_ncg - 10;
2327         }
2328 
2329         for (; cylno < sblock.fs_ncg; cylno++)
2330                 awtfs(fsbtodb(&sblock, (uint64_t)cgsblock(&sblock, cylno)),
2331                     (int)sbsize, (char *)&sblock, SAVE);
2332 
2333         /*
2334          * Flush out all the AIO writes we've done.  It's not
2335          * necessary to do this explicitly, but it's the only
2336          * way to report any errors from those writes.
2337          */
2338         flush_writes();
2339 
2340         /*
2341          * set clean flag
2342          */
2343         if (grow)
2344                 sblock.fs_clean = grow_fs_clean;
2345         else
2346                 sblock.fs_clean = FSCLEAN;
2347         sblock.fs_time = mkfstime;
2348         sblock.fs_state = FSOKAY - sblock.fs_time;
2349         wtfs((diskaddr_t)(SBOFF / sectorsize), sbsize, (char *)&sblock);
2350         isbad = 0;
2351 
2352         if (fsync(fso) == -1) {
2353                 saverr = errno;
2354                 (void) fprintf(stderr,
2355                     gettext("mkfs: fsync failed on write disk: %s\n"),
2356                     strerror(saverr));
2357                 /* we're just cleaning up, so keep going */
2358         }
2359         if (close(fsi) == -1) {
2360                 saverr = errno;
2361                 (void) fprintf(stderr,
2362                     gettext("mkfs: close failed on read disk: %s\n"),
2363                     strerror(saverr));
2364                 /* we're just cleaning up, so keep going */
2365         }
2366         if (close(fso) == -1) {
2367                 saverr = errno;
2368                 (void) fprintf(stderr,
2369                     gettext("mkfs: close failed on write disk: %s\n"),
2370                     strerror(saverr));
2371                 /* we're just cleaning up, so keep going */
2372         }
2373         fsi = fso = -1;
2374 
2375 #ifndef STANDALONE
2376         lockexit(0);
2377 #endif
2378 
2379         return (0);
2380 }
2381 
2382 static diskaddr_t
2383 get_device_size(int fd)
2384 {
2385         struct dk_minfo disk_info;
2386 
2387         if ((ioctl(fd, DKIOCGMEDIAINFO, (caddr_t)&disk_info)) == -1)
2388                 return (0);
2389 
2390         return (disk_info.dki_capacity);
2391 }
2392 
2393 /*
2394  * Figure out how big the partition we're dealing with is.
2395  * The value returned is in disk blocks (sectors);
2396  */
2397 static diskaddr_t
2398 get_max_size(int fd)
2399 {
2400         struct extvtoc vtoc;
2401         dk_gpt_t *efi_vtoc;
2402         diskaddr_t      slicesize;
2403 
2404         int index = read_extvtoc(fd, &vtoc);
2405 
2406         if (index >= 0) {
2407                 label_type = LABEL_TYPE_VTOC;
2408         } else {
2409                 if (index == VT_ENOTSUP || index == VT_ERROR) {
2410                         /* it might be an EFI label */
2411                         index = efi_alloc_and_read(fd, &efi_vtoc);
2412                         label_type = LABEL_TYPE_EFI;
2413                 }
2414         }
2415 
2416         if (index < 0) {
2417                 /*
2418                  * Since both attempts to read the label failed, we're
2419                  * going to use DKIOCGMEDIAINFO to get device size.
2420                  */
2421 
2422                 label_type = LABEL_TYPE_OTHER;
2423                 slicesize = get_device_size(fd);
2424                 if (slicesize == 0) {
2425                         switch (index) {
2426                         case VT_ERROR:
2427                                 break;
2428                         case VT_EIO:
2429                                 errno = EIO;
2430                                 break;
2431                         case VT_EINVAL:
2432                                 errno = EINVAL;
2433                         }
2434                         perror(gettext("Can not determine partition size"));
2435                         lockexit(32);
2436                 }
2437         }
2438 
2439         if (label_type == LABEL_TYPE_EFI) {
2440                 slicesize = efi_vtoc->efi_parts[index].p_size;
2441                 efi_free(efi_vtoc);
2442         } else if (label_type == LABEL_TYPE_VTOC) {
2443                 /*
2444                  * In the vtoc struct, p_size is a 32-bit signed quantity.
2445                  * In the dk_gpt struct (efi's version of the vtoc), p_size
2446                  * is an unsigned 64-bit quantity.  By casting the vtoc's
2447                  * psize to an unsigned 32-bit quantity, it will be copied
2448                  * to 'slicesize' (an unsigned 64-bit diskaddr_t) without
2449                  * sign extension.
2450                  */
2451 
2452                 slicesize = (uint32_t)vtoc.v_part[index].p_size;
2453         }
2454 
2455         dprintf(("DeBuG get_max_size index = %d, p_size = %lld, dolimit = %d\n",
2456             index, slicesize, (slicesize > FS_MAX)));
2457 
2458         /*
2459          * The next line limits a UFS file system to the maximum
2460          * supported size.
2461          */
2462 
2463         if (slicesize > FS_MAX)
2464                 return (FS_MAX);
2465         return (slicesize);
2466 }
2467 
2468 static long
2469 get_max_track_size(int fd)
2470 {
2471         struct dk_cinfo ci;
2472         long track_size = -1;
2473 
2474         if (ioctl(fd, DKIOCINFO, &ci) == 0) {
2475                 track_size = ci.dki_maxtransfer * DEV_BSIZE;
2476         }
2477 
2478         if ((track_size < 0)) {
2479                 int     error = 0;
2480                 int     maxphys;
2481                 int     gotit = 0;
2482 
2483                 gotit = fsgetmaxphys(&maxphys, &error);
2484                 if (gotit) {
2485                         track_size = MIN(MB, maxphys);
2486                 } else {
2487                         (void) fprintf(stderr, gettext(
2488 "Warning: Could not get system value for maxphys. The value for\n"
2489 "maxcontig will default to 1MB.\n"));
2490                         track_size = MB;
2491                 }
2492         }
2493         return (track_size);
2494 }
2495 
2496 /*
2497  * Initialize a cylinder group.
2498  */
2499 static void
2500 initcg(int cylno)
2501 {
2502         diskaddr_t cbase, d;
2503         diskaddr_t dlower;      /* last data block before cg metadata */
2504         diskaddr_t dupper;      /* first data block after cg metadata */
2505         diskaddr_t dmax;
2506         int64_t i;
2507         struct csum *cs;
2508         struct dinode *inode_buffer;
2509         int size;
2510 
2511         /*
2512          * Variables used to store intermediate results as a part of
2513          * the internal implementation of the cbtocylno() macros.
2514          */
2515         diskaddr_t bno;         /* UFS block number (not sector number) */
2516         int     cbcylno;        /* current cylinder number */
2517         int     cbcylno_sect;   /* sector offset within cylinder */
2518         int     cbsect_incr;    /* amount to increment sector offset */
2519 
2520         /*
2521          * Variables used to store intermediate results as a part of
2522          * the internal implementation of the cbtorpos() macros.
2523          */
2524         short   *cgblks;        /* pointer to array of free blocks in cg */
2525         int     trackrpos;      /* tmp variable for rotation position */
2526         int     trackoff;       /* offset within a track */
2527         int     trackoff_incr;  /* amount to increment trackoff */
2528         int     rpos;           /* rotation position of current block */
2529         int     rpos_incr;      /* amount to increment rpos per block */
2530 
2531         union cgun *icgun;      /* local pointer to a cg summary block */
2532 #define icg     (icgun->cg)
2533 
2534         icgun = (union cgun *)getbuf(&cgsumbuf, sizeof (union cgun));
2535 
2536         /*
2537          * Determine block bounds for cylinder group.
2538          * Allow space for super block summary information in first
2539          * cylinder group.
2540          */
2541         cbase = cgbase(&sblock, cylno);
2542         dmax = cbase + sblock.fs_fpg;
2543         if (dmax > sblock.fs_size)   /* last cg may be smaller than normal */
2544                 dmax = sblock.fs_size;
2545         dlower = cgsblock(&sblock, cylno) - cbase;
2546         dupper = cgdmin(&sblock, cylno) - cbase;
2547         if (cylno == 0)
2548                 dupper += howmany(sblock.fs_cssize, sblock.fs_fsize);
2549         cs = fscs + cylno;
2550         icg.cg_time = mkfstime;
2551         icg.cg_magic = CG_MAGIC;
2552         icg.cg_cgx = cylno;
2553         /* last one gets whatever's left */
2554         if (cylno == sblock.fs_ncg - 1)
2555                 icg.cg_ncyl = sblock.fs_ncyl - (sblock.fs_cpg * cylno);
2556         else
2557                 icg.cg_ncyl = sblock.fs_cpg;
2558         icg.cg_niblk = sblock.fs_ipg;
2559         icg.cg_ndblk = dmax - cbase;
2560         icg.cg_cs.cs_ndir = 0;
2561         icg.cg_cs.cs_nffree = 0;
2562         icg.cg_cs.cs_nbfree = 0;
2563         icg.cg_cs.cs_nifree = 0;
2564         icg.cg_rotor = 0;
2565         icg.cg_frotor = 0;
2566         icg.cg_irotor = 0;
2567         icg.cg_btotoff = &icg.cg_space[0] - (uchar_t *)(&icg.cg_link);
2568         icg.cg_boff = icg.cg_btotoff + sblock.fs_cpg * sizeof (long);
2569         icg.cg_iusedoff = icg.cg_boff +
2570             sblock.fs_cpg * sblock.fs_nrpos * sizeof (short);
2571         icg.cg_freeoff = icg.cg_iusedoff + howmany(sblock.fs_ipg, NBBY);
2572         icg.cg_nextfreeoff = icg.cg_freeoff +
2573             howmany(sblock.fs_cpg * sblock.fs_spc / NSPF(&sblock), NBBY);
2574         for (i = 0; i < sblock.fs_frag; i++) {
2575                 icg.cg_frsum[i] = 0;
2576         }
2577         bzero((caddr_t)cg_inosused(&icg), icg.cg_freeoff - icg.cg_iusedoff);
2578         icg.cg_cs.cs_nifree += sblock.fs_ipg;
2579         if (cylno == 0)
2580                 for (i = 0; i < UFSROOTINO; i++) {
2581                         setbit(cg_inosused(&icg), i);
2582                         icg.cg_cs.cs_nifree--;
2583                 }
2584 
2585         /*
2586          * Initialize all the inodes in the cylinder group using
2587          * random numbers.
2588          */
2589         size = sblock.fs_ipg * sizeof (struct dinode);
2590         inode_buffer = (struct dinode *)getbuf(&inodebuf, size);
2591 
2592         for (i = 0; i < sblock.fs_ipg; i++) {
2593                 IRANDOMIZE(&(inode_buffer[i].di_ic));
2594         }
2595 
2596         /*
2597          * Write all inodes in a single write for performance.
2598          */
2599         awtfs(fsbtodb(&sblock, (uint64_t)cgimin(&sblock, cylno)), (int)size,
2600             (char *)inode_buffer, RELEASE);
2601 
2602         bzero((caddr_t)cg_blktot(&icg), icg.cg_boff - icg.cg_btotoff);
2603         bzero((caddr_t)cg_blks(&sblock, &icg, 0),
2604             icg.cg_iusedoff - icg.cg_boff);
2605         bzero((caddr_t)cg_blksfree(&icg), icg.cg_nextfreeoff - icg.cg_freeoff);
2606 
2607         if (cylno > 0) {
2608                 for (d = 0; d < dlower; d += sblock.fs_frag) {
2609                         setblock(&sblock, cg_blksfree(&icg), d/sblock.fs_frag);
2610                         icg.cg_cs.cs_nbfree++;
2611                         cg_blktot(&icg)[cbtocylno(&sblock, d)]++;
2612                         cg_blks(&sblock, &icg, cbtocylno(&sblock, d))
2613                             [cbtorpos(&sblock, d)]++;
2614                 }
2615                 sblock.fs_dsize += dlower;
2616         }
2617         sblock.fs_dsize += icg.cg_ndblk - dupper;
2618         if ((i = dupper % sblock.fs_frag) != 0) {
2619                 icg.cg_frsum[sblock.fs_frag - i]++;
2620                 for (d = dupper + sblock.fs_frag - i; dupper < d; dupper++) {
2621                         setbit(cg_blksfree(&icg), dupper);
2622                         icg.cg_cs.cs_nffree++;
2623                 }
2624         }
2625 
2626         /*
2627          * WARNING: The following code is somewhat confusing, but
2628          * results in a substantial performance improvement in mkfs.
2629          *
2630          * Instead of using cbtocylno() and cbtorpos() macros, we
2631          * keep track of all the intermediate state of those macros
2632          * in some variables.  This allows simple addition to be
2633          * done to calculate the results as we step through the
2634          * blocks in an orderly fashion instead of the slower
2635          * multiplication and division the macros are forced to
2636          * used so they can support random input.  (Multiplication,
2637          * division, and remainder operations typically take about
2638          * 10x as many processor cycles as other operations.)
2639          *
2640          * The basic idea is to take code:
2641          *
2642          *      for (x = starting_x; x < max; x++)
2643          *              y = (x * c) / z
2644          *
2645          * and rewrite it to take advantage of the fact that
2646          * the variable x is incrementing in an orderly way:
2647          *
2648          *      intermediate = starting_x * c
2649          *      yval = intermediate / z
2650          *      for (x = starting_x; x < max; x++) {
2651          *              y = yval;
2652          *              intermediate += c
2653          *              if (intermediate > z) {
2654          *                      yval++;
2655          *                      intermediate -= z
2656          *              }
2657          *      }
2658          *
2659          * Performance has improved as much as 4X using this code.
2660          */
2661 
2662         /*
2663          * Initialize the starting points for all the cbtocylno()
2664          * macro variables and figure out the increments needed each
2665          * time through the loop.
2666          */
2667         cbcylno_sect = dupper * NSPF(&sblock);
2668         cbsect_incr = sblock.fs_frag * NSPF(&sblock);
2669         cbcylno = cbcylno_sect / sblock.fs_spc;
2670         cbcylno_sect %= sblock.fs_spc;
2671         cgblks = cg_blks(&sblock, &icg, cbcylno);
2672         bno = dupper / sblock.fs_frag;
2673 
2674         /*
2675          * Initialize the starting points for all the cbtorpos()
2676          * macro variables and figure out the increments needed each
2677          * time through the loop.
2678          *
2679          * It's harder to simplify the cbtorpos() macro if there were
2680          * alternate sectors specified (or if they previously existed
2681          * in the growfs case).  Since this is rare, we just revert to
2682          * using the macros in this case and skip the variable setup.
2683          */
2684         if (!spc_flag) {
2685                 trackrpos = (cbcylno_sect % sblock.fs_nsect) * sblock.fs_nrpos;
2686                 rpos = trackrpos / sblock.fs_nsect;
2687                 trackoff = trackrpos % sblock.fs_nsect;
2688                 trackoff_incr = cbsect_incr * sblock.fs_nrpos;
2689                 rpos_incr = (trackoff_incr / sblock.fs_nsect) % sblock.fs_nrpos;
2690                 trackoff_incr = trackoff_incr % sblock.fs_nsect;
2691         }
2692 
2693         /*
2694          * Loop through all the blocks, marking them free and
2695          * updating totals kept in the superblock and cg summary.
2696          */
2697         for (d = dupper; d + sblock.fs_frag <= dmax - cbase; ) {
2698                 setblock(&sblock, cg_blksfree(&icg),  bno);
2699                 icg.cg_cs.cs_nbfree++;
2700 
2701                 cg_blktot(&icg)[cbcylno]++;
2702 
2703                 if (!spc_flag)
2704                         cgblks[rpos]++;
2705                 else
2706                         cg_blks(&sblock, &icg, cbtocylno(&sblock, d))
2707                             [cbtorpos(&sblock, d)]++;
2708 
2709                 d += sblock.fs_frag;
2710                 bno++;
2711 
2712                 /*
2713                  * Increment the sector offset within the cylinder
2714                  * for the cbtocylno() macro reimplementation.  If
2715                  * we're beyond the end of the cylinder, update the
2716                  * cylinder number, calculate the offset in the
2717                  * new cylinder, and update the cgblks pointer
2718                  * to the next rotational position.
2719                  */
2720                 cbcylno_sect += cbsect_incr;
2721                 if (cbcylno_sect >= sblock.fs_spc) {
2722                         cbcylno++;
2723                         cbcylno_sect -= sblock.fs_spc;
2724                         cgblks += sblock.fs_nrpos;
2725                 }
2726 
2727                 /*
2728                  * If there aren't alternate sectors, increment the
2729                  * rotational position variables for the cbtorpos()
2730                  * reimplementation.  Note that we potentially
2731                  * increment rpos twice.  Once by rpos_incr, and one
2732                  * more time when we wrap to a new track because
2733                  * trackoff >= fs_nsect.
2734                  */
2735                 if (!spc_flag) {
2736                         trackoff += trackoff_incr;
2737                         rpos += rpos_incr;
2738                         if (trackoff >= sblock.fs_nsect) {
2739                                 trackoff -= sblock.fs_nsect;
2740                                 rpos++;
2741                         }
2742                         if (rpos >= sblock.fs_nrpos)
2743                                 rpos -= sblock.fs_nrpos;
2744                 }
2745         }
2746 
2747         if (d < dmax - cbase) {
2748                 icg.cg_frsum[dmax - cbase - d]++;
2749                 for (; d < dmax - cbase; d++) {
2750                         setbit(cg_blksfree(&icg), d);
2751                         icg.cg_cs.cs_nffree++;
2752                 }
2753         }
2754         sblock.fs_cstotal.cs_ndir += icg.cg_cs.cs_ndir;
2755         sblock.fs_cstotal.cs_nffree += icg.cg_cs.cs_nffree;
2756         sblock.fs_cstotal.cs_nbfree += icg.cg_cs.cs_nbfree;
2757         sblock.fs_cstotal.cs_nifree += icg.cg_cs.cs_nifree;
2758         *cs = icg.cg_cs;
2759         awtfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, cylno)),
2760             sblock.fs_bsize, (char *)&icg, RELEASE);
2761 }
2762 
2763 /*
2764  * initialize the file system
2765  */
2766 struct inode node;
2767 
2768 #define LOSTDIR
2769 #ifdef LOSTDIR
2770 #define PREDEFDIR 3
2771 #else
2772 #define PREDEFDIR 2
2773 #endif
2774 
2775 struct direct root_dir[] = {
2776         { UFSROOTINO, sizeof (struct direct), 1, "." },
2777         { UFSROOTINO, sizeof (struct direct), 2, ".." },
2778 #ifdef LOSTDIR
2779         { LOSTFOUNDINO, sizeof (struct direct), 10, "lost+found" },
2780 #endif
2781 };
2782 #ifdef LOSTDIR
2783 struct direct lost_found_dir[] = {
2784         { LOSTFOUNDINO, sizeof (struct direct), 1, "." },
2785         { UFSROOTINO, sizeof (struct direct), 2, ".." },
2786         { 0, DIRBLKSIZ, 0, 0 },
2787 };
2788 #endif
2789 char buf[MAXBSIZE];
2790 
2791 static void
2792 fsinit()
2793 {
2794         int i;
2795 
2796 
2797         /*
2798          * initialize the node
2799          */
2800         node.i_atime = mkfstime;
2801         node.i_mtime = mkfstime;
2802         node.i_ctime = mkfstime;
2803 #ifdef LOSTDIR
2804         /*
2805          * create the lost+found directory
2806          */
2807         (void) makedir(lost_found_dir, 2);
2808         for (i = DIRBLKSIZ; i < sblock.fs_bsize; i += DIRBLKSIZ) {
2809                 bcopy(&lost_found_dir[2], &buf[i], DIRSIZ(&lost_found_dir[2]));
2810         }
2811         node.i_number = LOSTFOUNDINO;
2812         node.i_smode = IFDIR | 0700;
2813         node.i_nlink = 2;
2814         node.i_size = sblock.fs_bsize;
2815         node.i_db[0] = alloc((int)node.i_size, node.i_mode);
2816         node.i_blocks = btodb(fragroundup(&sblock, (int)node.i_size));
2817         IRANDOMIZE(&node.i_ic);
2818         wtfs(fsbtodb(&sblock, (uint64_t)node.i_db[0]), (int)node.i_size, buf);
2819         iput(&node);
2820 #endif
2821         /*
2822          * create the root directory
2823          */
2824         node.i_number = UFSROOTINO;
2825         node.i_mode = IFDIR | UMASK;
2826         node.i_nlink = PREDEFDIR;
2827         node.i_size = makedir(root_dir, PREDEFDIR);
2828         node.i_db[0] = alloc(sblock.fs_fsize, node.i_mode);
2829         /* i_size < 2GB because we are initializing the file system */
2830         node.i_blocks = btodb(fragroundup(&sblock, (int)node.i_size));
2831         IRANDOMIZE(&node.i_ic);
2832         wtfs(fsbtodb(&sblock, (uint64_t)node.i_db[0]), sblock.fs_fsize, buf);
2833         iput(&node);
2834 }
2835 
2836 /*
2837  * construct a set of directory entries in "buf".
2838  * return size of directory.
2839  */
2840 static int
2841 makedir(struct direct *protodir, int entries)
2842 {
2843         char *cp;
2844         int i;
2845         ushort_t spcleft;
2846 
2847         spcleft = DIRBLKSIZ;
2848         for (cp = buf, i = 0; i < entries - 1; i++) {
2849                 protodir[i].d_reclen = DIRSIZ(&protodir[i]);
2850                 bcopy(&protodir[i], cp, protodir[i].d_reclen);
2851                 cp += protodir[i].d_reclen;
2852                 spcleft -= protodir[i].d_reclen;
2853         }
2854         protodir[i].d_reclen = spcleft;
2855         bcopy(&protodir[i], cp, DIRSIZ(&protodir[i]));
2856         return (DIRBLKSIZ);
2857 }
2858 
2859 /*
2860  * allocate a block or frag
2861  */
2862 static daddr32_t
2863 alloc(int size, int mode)
2864 {
2865         int i, frag;
2866         daddr32_t d;
2867 
2868         rdfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, 0)), sblock.fs_cgsize,
2869             (char *)&acg);
2870         if (acg.cg_magic != CG_MAGIC) {
2871                 (void) fprintf(stderr, gettext("cg 0: bad magic number\n"));
2872                 lockexit(32);
2873         }
2874         if (acg.cg_cs.cs_nbfree == 0) {
2875                 (void) fprintf(stderr,
2876                     gettext("first cylinder group ran out of space\n"));
2877                 lockexit(32);
2878         }
2879         for (d = 0; d < acg.cg_ndblk; d += sblock.fs_frag)
2880                 if (isblock(&sblock, cg_blksfree(&acg), d / sblock.fs_frag))
2881                         goto goth;
2882         (void) fprintf(stderr,
2883             gettext("internal error: can't find block in cyl 0\n"));
2884         lockexit(32);
2885 goth:
2886         clrblock(&sblock, cg_blksfree(&acg), d / sblock.fs_frag);
2887         acg.cg_cs.cs_nbfree--;
2888         sblock.fs_cstotal.cs_nbfree--;
2889         fscs[0].cs_nbfree--;
2890         if (mode & IFDIR) {
2891                 acg.cg_cs.cs_ndir++;
2892                 sblock.fs_cstotal.cs_ndir++;
2893                 fscs[0].cs_ndir++;
2894         }
2895         cg_blktot(&acg)[cbtocylno(&sblock, d)]--;
2896         cg_blks(&sblock, &acg, cbtocylno(&sblock, d))[cbtorpos(&sblock, d)]--;
2897         if (size != sblock.fs_bsize) {
2898                 frag = howmany(size, sblock.fs_fsize);
2899                 fscs[0].cs_nffree += sblock.fs_frag - frag;
2900                 sblock.fs_cstotal.cs_nffree += sblock.fs_frag - frag;
2901                 acg.cg_cs.cs_nffree += sblock.fs_frag - frag;
2902                 acg.cg_frsum[sblock.fs_frag - frag]++;
2903                 for (i = frag; i < sblock.fs_frag; i++)
2904                         setbit(cg_blksfree(&acg), d + i);
2905         }
2906         wtfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, 0)), sblock.fs_cgsize,
2907             (char *)&acg);
2908         return (d);
2909 }
2910 
2911 /*
2912  * Allocate an inode on the disk
2913  */
2914 static void
2915 iput(struct inode *ip)
2916 {
2917         struct dinode buf[MAXINOPB];
2918         diskaddr_t d;
2919 
2920         rdfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, 0)), sblock.fs_cgsize,
2921             (char *)&acg);
2922         if (acg.cg_magic != CG_MAGIC) {
2923                 (void) fprintf(stderr, gettext("cg 0: bad magic number\n"));
2924                 lockexit(32);
2925         }
2926         acg.cg_cs.cs_nifree--;
2927         setbit(cg_inosused(&acg), ip->i_number);
2928         wtfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, 0)), sblock.fs_cgsize,
2929             (char *)&acg);
2930         sblock.fs_cstotal.cs_nifree--;
2931         fscs[0].cs_nifree--;
2932         if ((int)ip->i_number >= sblock.fs_ipg * sblock.fs_ncg) {
2933                 (void) fprintf(stderr,
2934                     gettext("fsinit: inode value out of range (%d).\n"),
2935                     ip->i_number);
2936                 lockexit(32);
2937         }
2938         d = fsbtodb(&sblock, (uint64_t)itod(&sblock, (int)ip->i_number));
2939         rdfs(d, sblock.fs_bsize, (char *)buf);
2940         buf[itoo(&sblock, (int)ip->i_number)].di_ic = ip->i_ic;
2941         wtfs(d, sblock.fs_bsize, (char *)buf);
2942 }
2943 
2944 /*
2945  * getbuf()     -- Get a buffer for use in an AIO operation.  Buffer
2946  *              is zero'd the first time returned, left with whatever
2947  *              was in memory after that.  This function actually gets
2948  *              enough memory the first time it's called to support
2949  *              MAXBUF buffers like a slab allocator.  When all the
2950  *              buffers are in use, it waits for an aio to complete
2951  *              and make a buffer available.
2952  *
2953  *              Never returns an error.  Either succeeds or exits.
2954  */
2955 static char *
2956 getbuf(bufhdr *bufhead, int size)
2957 {
2958         bufhdr *pbuf;
2959         bufhdr *prev;
2960         int i;
2961         int buf_size, max_bufs;
2962 
2963         /*
2964          * Initialize all the buffers
2965          */
2966         if (bufhead->head == NULL) {
2967                 /*
2968                  * round up the size of our buffer header to a
2969                  * 16 byte boundary so the address we return to
2970                  * the caller is "suitably aligned".
2971                  */
2972                 bufhdrsize = (sizeof (bufhdr) + 15) & ~15;
2973 
2974                 /*
2975                  * Add in our header to the buffer and round it all up to
2976                  * a 16 byte boundry so each member of the slab is aligned.
2977                  */
2978                 buf_size = (size + bufhdrsize + 15) & ~15;
2979 
2980                 /*
2981                  * Limit number of buffers to lesser of MAXBUFMEM's worth
2982                  * or MAXBUF, whichever is less.
2983                  */
2984                 max_bufs = MAXBUFMEM / buf_size;
2985                 if (max_bufs > MAXBUF)
2986                         max_bufs = MAXBUF;
2987 
2988                 pbuf = (bufhdr *)calloc(max_bufs, buf_size);
2989                 if (pbuf == NULL) {
2990                         perror("calloc");
2991                         lockexit(32);
2992                 }
2993 
2994                 bufhead->head = bufhead;
2995                 prev = bufhead;
2996                 for (i = 0; i < max_bufs; i++) {
2997                         pbuf->head = bufhead;
2998                         prev->next = pbuf;
2999                         prev = pbuf;
3000                         pbuf = (bufhdr *)((char *)pbuf + buf_size);
3001                 }
3002         }
3003 
3004         /*
3005          * Get an available buffer, waiting for I/O if necessary
3006          */
3007         wait_for_write(NOBLOCK);
3008         while (bufhead->next == NULL)
3009                 wait_for_write(BLOCK);
3010 
3011         /*
3012          * Take the buffer off the list
3013          */
3014         pbuf = bufhead->next;
3015         bufhead->next = pbuf->next;
3016         pbuf->next = NULL;
3017 
3018         /*
3019          * return the empty buffer space just past the header
3020          */
3021         return ((char *)pbuf + bufhdrsize);
3022 }
3023 
3024 /*
3025  * freebuf()    -- Free a buffer gotten previously through getbuf.
3026  *              Puts the buffer back on the appropriate list for
3027  *              later use.  Never calls free().
3028  *
3029  * Assumes that SIGINT is blocked.
3030  */
3031 static void
3032 freebuf(char *buf)
3033 {
3034         bufhdr *pbuf;
3035         bufhdr *bufhead;
3036 
3037         /*
3038          * get the header for this buffer
3039          */
3040         pbuf = (bufhdr *)(buf - bufhdrsize);
3041 
3042         /*
3043          * Put it back on the list of available buffers
3044          */
3045         bufhead = pbuf->head;
3046         pbuf->next = bufhead->next;
3047         bufhead->next = pbuf;
3048 }
3049 
3050 /*
3051  * freetrans()  -- Free a transaction gotten previously through getaiop.
3052  *              Puts the transaction struct back on the appropriate list for
3053  *              later use.  Never calls free().
3054  *
3055  * Assumes that SIGINT is blocked.
3056  */
3057 static void
3058 freetrans(aio_trans *transp)
3059 {
3060         /*
3061          * free the buffer associated with this AIO if needed
3062          */
3063         if (transp->release == RELEASE)
3064                 freebuf(transp->buffer);
3065 
3066         /*
3067          * Put transaction on the free list
3068          */
3069         transp->next = results.trans;
3070         results.trans = transp;
3071 }
3072 
3073 /*
3074  * wait_for_write()     -- Wait for an aio write to complete.  Return
3075  *                      the transaction structure for that write.
3076  *
3077  * Blocks SIGINT if necessary.
3078  */
3079 aio_trans *
3080 wait_for_write(int block)
3081 {
3082         aio_trans       *transp;
3083         aio_result_t    *resultp;
3084         static struct timeval  zero_wait = { 0, 0 };
3085         sigset_t        old_mask;
3086 
3087         /*
3088          * If we know there aren't any outstanding transactions, just return
3089          */
3090         if (results.outstanding == 0)
3091                 return ((aio_trans *) 0);
3092 
3093         block_sigint(&old_mask);
3094 
3095         resultp = aiowait(block ? NULL : &zero_wait);
3096         if (resultp == NULL ||
3097             (resultp == (aio_result_t *)-1 && errno == EINVAL)) {
3098                 unblock_sigint(&old_mask);
3099                 return ((aio_trans *) 0);
3100         }
3101 
3102         results.outstanding--;
3103         transp = (aio_trans *)resultp;
3104 
3105         if (resultp->aio_return != transp->size) {
3106                 if (resultp->aio_return == -1) {
3107                         /*
3108                          * The aiowrite() may have failed because the
3109                          * kernel didn't have enough memory to do the job.
3110                          * Flush all pending writes and try a normal
3111                          * write().  wtfs_breakup() will call exit if it
3112                          * fails, so we don't worry about errors here.
3113                          */
3114                         flush_writes();
3115                         wtfs_breakup(transp->bno, transp->size, transp->buffer);
3116                 } else {
3117                         (void) fprintf(stderr, gettext(
3118                             "short write (%d of %d bytes) on sector %lld\n"),
3119                             resultp->aio_return, transp->size,
3120                             transp->bno);
3121                         /*
3122                          * Don't unblock SIGINT, to avoid potential
3123                          * looping due to queued interrupts and
3124                          * error handling.
3125                          */
3126                         lockexit(32);
3127                 }
3128         }
3129 
3130         resultp->aio_return = 0;
3131         freetrans(transp);
3132         unblock_sigint(&old_mask);
3133         return (transp);
3134 }
3135 
3136 /*
3137  * flush_writes()       -- flush all the outstanding aio writes.
3138  */
3139 static void
3140 flush_writes(void)
3141 {
3142         while (wait_for_write(BLOCK))
3143                 ;
3144 }
3145 
3146 /*
3147  * get_aiop()   -- find and return an aio_trans structure on which a new
3148  *              aio can be done.  Blocks on aiowait() if needed.  Reaps
3149  *              all outstanding completed aio's.
3150  *
3151  * Assumes that SIGINT is blocked.
3152  */
3153 aio_trans *
3154 get_aiop()
3155 {
3156         int i;
3157         aio_trans *transp;
3158         aio_trans *prev;
3159 
3160         /*
3161          * initialize aio stuff
3162          */
3163         if (!aio_inited) {
3164                 aio_inited = 1;
3165 
3166                 results.maxpend = 0;
3167                 results.outstanding = 0;
3168                 results.max = MAXAIO;
3169 
3170                 results.trans = (aio_trans *)calloc(results.max,
3171                     sizeof (aio_trans));
3172                 if (results.trans == NULL) {
3173                         perror("calloc");
3174                         lockexit(32);
3175                 }
3176 
3177                 /*
3178                  * Initialize the linked list of aio transaction
3179                  * structures.  Note that the final "next" pointer
3180                  * will be NULL since we got the buffer from calloc().
3181                  */
3182                 prev = results.trans;
3183                 for (i = 1; i < results.max; i++) {
3184                         prev->next = &(results.trans[i]);
3185                         prev = prev->next;
3186                 }
3187         }
3188 
3189         wait_for_write(NOBLOCK);
3190         while (results.trans == NULL)
3191                 wait_for_write(BLOCK);
3192         transp = results.trans;
3193         results.trans = results.trans->next;
3194 
3195         transp->next = 0;
3196         transp->resultbuf.aio_return = AIO_INPROGRESS;
3197         return (transp);
3198 }
3199 
3200 /*
3201  * read a block from the file system
3202  */
3203 static void
3204 rdfs(diskaddr_t bno, int size, char *bf)
3205 {
3206         int n, saverr;
3207 
3208         /*
3209          * In case we need any data that's pending in an aiowrite(),
3210          * we wait for them all to complete before doing a read.
3211          */
3212         flush_writes();
3213 
3214         /*
3215          * Note: the llseek() can succeed, even if the offset is out of range.
3216          * It's not until the file i/o operation (the read()) that one knows
3217          * for sure if the raw device can handle the offset.
3218          */
3219         if (llseek(fsi, (offset_t)bno * sectorsize, 0) < 0) {
3220                 saverr = errno;
3221                 (void) fprintf(stderr,
3222                     gettext("seek error on sector %lld: %s\n"),
3223                     bno, strerror(saverr));
3224                 lockexit(32);
3225         }
3226         n = read(fsi, bf, size);
3227         if (n != size) {
3228                 saverr = errno;
3229                 if (n == -1)
3230                         (void) fprintf(stderr,
3231                             gettext("read error on sector %lld: %s\n"),
3232                             bno, strerror(saverr));
3233                 else
3234                         (void) fprintf(stderr, gettext(
3235                             "short read (%d of %d bytes) on sector %lld\n"),
3236                             n, size, bno);
3237                 lockexit(32);
3238         }
3239 }
3240 
3241 /*
3242  * write a block to the file system
3243  */
3244 static void
3245 wtfs(diskaddr_t bno, int size, char *bf)
3246 {
3247         int n, saverr;
3248 
3249         if (fso == -1)
3250                 return;
3251 
3252         /*
3253          * Note: the llseek() can succeed, even if the offset is out of range.
3254          * It's not until the file i/o operation (the write()) that one knows
3255          * for sure if the raw device can handle the offset.
3256          */
3257         if (llseek(fso, (offset_t)bno * sectorsize, 0) < 0) {
3258                 saverr = errno;
3259                 (void) fprintf(stderr,
3260                     gettext("seek error on sector %lld: %s\n"),
3261                     bno, strerror(saverr));
3262                 lockexit(32);
3263         }
3264         if (Nflag)
3265                 return;
3266         n = write(fso, bf, size);
3267         if (n != size) {
3268                 saverr = errno;
3269                 if (n == -1)
3270                         (void) fprintf(stderr,
3271                             gettext("write error on sector %lld: %s\n"),
3272                             bno, strerror(saverr));
3273                 else
3274                         (void) fprintf(stderr, gettext(
3275                             "short write (%d of %d bytes) on sector %lld\n"),
3276                             n, size, bno);
3277                 lockexit(32);
3278         }
3279 }
3280 
3281 /*
3282  * write a block to the file system -- buffered with aio
3283  */
3284 static void
3285 awtfs(diskaddr_t bno, int size, char *bf, int release)
3286 {
3287         int n;
3288         aio_trans       *transp;
3289         sigset_t        old_mask;
3290 
3291         if (fso == -1)
3292                 return;
3293 
3294         /*
3295          * We need to keep things consistent if we get interrupted,
3296          * so defer any expected interrupts for the time being.
3297          */
3298         block_sigint(&old_mask);
3299 
3300         if (Nflag) {
3301                 if (release == RELEASE)
3302                         freebuf(bf);
3303         } else {
3304                 transp = get_aiop();
3305                 transp->bno = bno;
3306                 transp->buffer = bf;
3307                 transp->size = size;
3308                 transp->release = release;
3309 
3310                 n = aiowrite(fso, bf, size, (off_t)bno * sectorsize,
3311                     SEEK_SET, &transp->resultbuf);
3312 
3313                 if (n < 0) {
3314                         /*
3315                          * The aiowrite() may have failed because the
3316                          * kernel didn't have enough memory to do the job.
3317                          * Flush all pending writes and try a normal
3318                          * write().  wtfs_breakup() will call exit if it
3319                          * fails, so we don't worry about errors here.
3320                          */
3321                         flush_writes();
3322                         wtfs_breakup(transp->bno, transp->size, transp->buffer);
3323                         freetrans(transp);
3324                 } else {
3325                         /*
3326                          * Keep track of our pending writes.
3327                          */
3328                         results.outstanding++;
3329                         if (results.outstanding > results.maxpend)
3330                                 results.maxpend = results.outstanding;
3331                 }
3332         }
3333 
3334         unblock_sigint(&old_mask);
3335 }
3336 
3337 
3338 /*
3339  * write a block to the file system, but break it up into sbsize
3340  * chunks to avoid forcing a large amount of memory to be locked down.
3341  * Only used as a fallback when an aio write has failed.
3342  */
3343 static void
3344 wtfs_breakup(diskaddr_t bno, int size, char *bf)
3345 {
3346         int n, saverr;
3347         int wsize;
3348         int block_incr = sbsize / sectorsize;
3349 
3350         if (size < sbsize)
3351                 wsize = size;
3352         else
3353                 wsize = sbsize;
3354 
3355         n = 0;
3356         while (size) {
3357                 /*
3358                  * Note: the llseek() can succeed, even if the offset is
3359                  * out of range.  It's not until the file i/o operation
3360                  * (the write()) that one knows for sure if the raw device
3361                  * can handle the offset.
3362                  */
3363                 if (llseek(fso, (offset_t)bno * sectorsize, 0) < 0) {
3364                         saverr = errno;
3365                         (void) fprintf(stderr,
3366                             gettext("seek error on sector %lld: %s\n"),
3367                             bno, strerror(saverr));
3368                         lockexit(32);
3369                 }
3370 
3371                 n = write(fso, bf, wsize);
3372                 if (n == -1) {
3373                         saverr = errno;
3374                         (void) fprintf(stderr,
3375                             gettext("write error on sector %lld: %s\n"),
3376                             bno, strerror(saverr));
3377                         lockexit(32);
3378                 }
3379                 if (n != wsize) {
3380                         saverr = errno;
3381                         (void) fprintf(stderr, gettext(
3382                             "short write (%d of %d bytes) on sector %lld\n"),
3383                             n, size, bno);
3384                         lockexit(32);
3385                 }
3386 
3387                 bno += block_incr;
3388                 bf += wsize;
3389                 size -= wsize;
3390                 if (size < wsize)
3391                         wsize = size;
3392         }
3393 }
3394 
3395 
3396 /*
3397  * check if a block is available
3398  */
3399 static int
3400 isblock(struct fs *fs, unsigned char *cp, int h)
3401 {
3402         unsigned char mask;
3403 
3404         switch (fs->fs_frag) {
3405         case 8:
3406                 return (cp[h] == 0xff);
3407         case 4:
3408                 mask = 0x0f << ((h & 0x1) << 2);
3409                 return ((cp[h >> 1] & mask) == mask);
3410         case 2:
3411                 mask = 0x03 << ((h & 0x3) << 1);
3412                 return ((cp[h >> 2] & mask) == mask);
3413         case 1:
3414                 mask = 0x01 << (h & 0x7);
3415                 return ((cp[h >> 3] & mask) == mask);
3416         default:
3417                 (void) fprintf(stderr, "isblock bad fs_frag %d\n", fs->fs_frag);
3418                 return (0);
3419         }
3420 }
3421 
3422 /*
3423  * take a block out of the map
3424  */
3425 static void
3426 clrblock(struct fs *fs, unsigned char *cp, int h)
3427 {
3428         switch ((fs)->fs_frag) {
3429         case 8:
3430                 cp[h] = 0;
3431                 return;
3432         case 4:
3433                 cp[h >> 1] &= ~(0x0f << ((h & 0x1) << 2));
3434                 return;
3435         case 2:
3436                 cp[h >> 2] &= ~(0x03 << ((h & 0x3) << 1));
3437                 return;
3438         case 1:
3439                 cp[h >> 3] &= ~(0x01 << (h & 0x7));
3440                 return;
3441         default:
3442                 (void) fprintf(stderr,
3443                     gettext("clrblock: bad fs_frag value %d\n"), fs->fs_frag);
3444                 return;
3445         }
3446 }
3447 
3448 /*
3449  * put a block into the map
3450  */
3451 static void
3452 setblock(struct fs *fs, unsigned char *cp, int h)
3453 {
3454         switch (fs->fs_frag) {
3455         case 8:
3456                 cp[h] = 0xff;
3457                 return;
3458         case 4:
3459                 cp[h >> 1] |= (0x0f << ((h & 0x1) << 2));
3460                 return;
3461         case 2:
3462                 cp[h >> 2] |= (0x03 << ((h & 0x3) << 1));
3463                 return;
3464         case 1:
3465                 cp[h >> 3] |= (0x01 << (h & 0x7));
3466                 return;
3467         default:
3468                 (void) fprintf(stderr,
3469                     gettext("setblock: bad fs_frag value %d\n"), fs->fs_frag);
3470                 return;
3471         }
3472 }
3473 
3474 static void
3475 usage(void)
3476 {
3477         (void) fprintf(stderr,
3478             gettext("ufs usage: mkfs [-F FSType] [-V] [-m] [-o options] "
3479             "special "                          /* param 0 */
3480             "size(sectors) \\ \n"));            /* param 1 */
3481         (void) fprintf(stderr,
3482             "[nsect "                           /* param 2 */
3483             "ntrack "                           /* param 3 */
3484             "bsize "                            /* param 4 */
3485             "fragsize "                         /* param 5 */
3486             "cpg "                              /* param 6 */
3487             "free "                             /* param 7 */
3488             "rps "                              /* param 8 */
3489             "nbpi "                             /* param 9 */
3490             "opt "                              /* param 10 */
3491             "apc "                              /* param 11 */
3492             "gap "                              /* param 12 */
3493             "nrpos "                            /* param 13 */
3494             "maxcontig "                        /* param 14 */
3495             "mtb]\n");                          /* param 15 */
3496         (void) fprintf(stderr,
3497             gettext(" -m : dump fs cmd line used to make this partition\n"
3498             " -V :print this command line and return\n"
3499             " -o :ufs options: :nsect=%d,ntrack=%d,bsize=%d,fragsize=%d\n"
3500             " -o :ufs options: :cgsize=%d,free=%d,rps=%d,nbpi=%d,opt=%c\n"
3501             " -o :ufs options: :apc=%d,gap=%d,nrpos=%d,maxcontig=%d\n"
3502             " -o :ufs options: :mtb=%c,calcsb,calcbinsb\n"
3503 "NOTE that all -o suboptions: must be separated only by commas so as to\n"
3504 "be parsed as a single argument\n"),
3505             nsect, ntrack, bsize, fragsize, cpg, sblock.fs_minfree, rps,
3506             nbpi, opt, apc, (rotdelay == -1) ? 0 : rotdelay,
3507             sblock.fs_nrpos, maxcontig, mtb);
3508         lockexit(32);
3509 }
3510 
3511 /*ARGSUSED*/
3512 static void
3513 dump_fscmd(char *fsys, int fsi)
3514 {
3515         int64_t used, bpcg, inospercg;
3516         int64_t nbpi;
3517         uint64_t nbytes64;
3518 
3519         bzero((char *)&sblock, sizeof (sblock));
3520         rdfs((diskaddr_t)SBLOCK, SBSIZE, (char *)&sblock);
3521 
3522         /*
3523          * ensure a valid file system and if not, exit with error or else
3524          * we will end up computing block numbers etc and dividing by zero
3525          * which will cause floating point errors in this routine.
3526          */
3527 
3528         if ((sblock.fs_magic != FS_MAGIC) &&
3529             (sblock.fs_magic != MTB_UFS_MAGIC)) {
3530                 (void) fprintf(stderr, gettext(
3531                     "[not currently a valid file system - bad superblock]\n"));
3532                 lockexit(32);
3533         }
3534 
3535         if (sblock.fs_magic == FS_MAGIC &&
3536             (sblock.fs_version != UFS_EFISTYLE4NONEFI_VERSION_2 &&
3537             sblock.fs_version != UFS_VERSION_MIN)) {
3538                 (void) fprintf(stderr, gettext(
3539                     "Unknown version of UFS format: %d\n"), sblock.fs_version);
3540                 lockexit(32);
3541         }
3542 
3543         if (sblock.fs_magic == MTB_UFS_MAGIC &&
3544             (sblock.fs_version > MTB_UFS_VERSION_1 ||
3545             sblock.fs_version < MTB_UFS_VERSION_MIN)) {
3546                 (void) fprintf(stderr, gettext(
3547                     "Unknown version of UFS format: %d\n"), sblock.fs_version);
3548                 lockexit(32);
3549         }
3550 
3551         /*
3552          * Compute a reasonable nbpi value.
3553          * The algorithm for "used" is copied from code
3554          * in main() verbatim.
3555          * The nbpi equation is taken from main where the
3556          * fs_ipg value is set for the last time.  The INOPB(...) - 1
3557          * is used to account for the roundup.
3558          * The problem is that a range of nbpi values map to
3559          * the same file system layout.  So it is not possible
3560          * to calculate the exact value specified when the file
3561          * system was created.  So instead we determine the top
3562          * end of the range of values.
3563          */
3564         bpcg = sblock.fs_spc * sectorsize;
3565         inospercg = (int64_t)roundup(bpcg / sizeof (struct dinode),
3566             INOPB(&sblock));
3567         if (inospercg > MAXIpG(&sblock))
3568                 inospercg = MAXIpG(&sblock);
3569         used = (int64_t)
3570             (sblock.fs_iblkno + inospercg / INOPF(&sblock)) * NSPF(&sblock);
3571         used *= sectorsize;
3572         nbytes64 = (uint64_t)sblock.fs_cpg * bpcg - used;
3573 
3574         /*
3575          * The top end of the range of values for nbpi may not be
3576          * a valid command line value for mkfs. Report the bottom
3577          * end instead.
3578          */
3579         nbpi = (int64_t)(nbytes64 / (sblock.fs_ipg));
3580 
3581         (void) fprintf(stdout, gettext("mkfs -F ufs -o "), fsys);
3582         (void) fprintf(stdout, "nsect=%d,ntrack=%d,",
3583             sblock.fs_nsect, sblock.fs_ntrak);
3584         (void) fprintf(stdout, "bsize=%d,fragsize=%d,cgsize=%d,free=%d,",
3585             sblock.fs_bsize, sblock.fs_fsize, sblock.fs_cpg, sblock.fs_minfree);
3586         (void) fprintf(stdout, "rps=%d,nbpi=%lld,opt=%c,apc=%d,gap=%d,",
3587             sblock.fs_rps, nbpi, (sblock.fs_optim == FS_OPTSPACE) ? 's' : 't',
3588             (sblock.fs_ntrak * sblock.fs_nsect) - sblock.fs_spc,
3589             sblock.fs_rotdelay);
3590         (void) fprintf(stdout, "nrpos=%d,maxcontig=%d,mtb=%c ",
3591             sblock.fs_nrpos, sblock.fs_maxcontig,
3592             ((sblock.fs_magic == MTB_UFS_MAGIC) ? 'y' : 'n'));
3593         (void) fprintf(stdout, "%s %lld\n", fsys,
3594             fsbtodb(&sblock, sblock.fs_size));
3595 
3596         bzero((char *)&sblock, sizeof (sblock));
3597 }
3598 
3599 /* number ************************************************************* */
3600 /*                                                                      */
3601 /* Convert a numeric string arg to binary                               */
3602 /*                                                                      */
3603 /* Args:        d_value - default value, if have parse error            */
3604 /*              param - the name of the argument, for error messages    */
3605 /*              flags - parser state and what's allowed in the arg      */
3606 /* Global arg:  string - pointer to command arg                         */
3607 /*                                                                      */
3608 /* Valid forms: 123 | 123k | 123*123 | 123x123                          */
3609 /*                                                                      */
3610 /* Return:      converted number                                        */
3611 /*                                                                      */
3612 /* ******************************************************************** */
3613 
3614 static uint64_t
3615 number(uint64_t d_value, char *param, int flags)
3616 {
3617         char *cs;
3618         uint64_t n, t;
3619         uint64_t cut = BIG / 10;    /* limit to avoid overflow */
3620         int minus = 0;
3621 
3622         cs = string;
3623         if (*cs == '-') {
3624                 minus = 1;
3625                 cs += 1;
3626         }
3627         if ((*cs < '0') || (*cs > '9')) {
3628                 goto bail_out;
3629         }
3630         n = 0;
3631         while ((*cs >= '0') && (*cs <= '9') && (n <= cut)) {
3632                 n = n*10 + *cs++ - '0';
3633         }
3634         if (minus)
3635                 n = -n;
3636         for (;;) {
3637                 switch (*cs++) {
3638                 case 'k':
3639                         if (flags & ALLOW_END_ONLY)
3640                                 goto bail_out;
3641                         if (n > (BIG / 1024))
3642                                 goto overflow;
3643                         n *= 1024;
3644                         continue;
3645 
3646                 case '*':
3647                 case 'x':
3648                         if (flags & ALLOW_END_ONLY)
3649                                 goto bail_out;
3650                         string = cs;
3651                         t = number(d_value, param, flags);
3652                         if (n > (BIG / t))
3653                                 goto overflow;
3654                         n *= t;
3655                         cs = string + 1; /* adjust for -- below */
3656 
3657                         /* recursion has read rest of expression */
3658                         /* FALLTHROUGH */
3659 
3660                 case ',':
3661                 case '\0':
3662                         cs--;
3663                         string = cs;
3664                         return (n);
3665 
3666                 case '%':
3667                         if (flags & ALLOW_END_ONLY)
3668                                 goto bail_out;
3669                         if (flags & ALLOW_PERCENT) {
3670                                 flags &= ~ALLOW_PERCENT;
3671                                 flags |= ALLOW_END_ONLY;
3672                                 continue;
3673                         }
3674                         goto bail_out;
3675 
3676                 case 'm':
3677                         if (flags & ALLOW_END_ONLY)
3678                                 goto bail_out;
3679                         if (flags & ALLOW_MS1) {
3680                                 flags &= ~ALLOW_MS1;
3681                                 flags |= ALLOW_MS2;
3682                                 continue;
3683                         }
3684                         goto bail_out;
3685 
3686                 case 's':
3687                         if (flags & ALLOW_END_ONLY)
3688                                 goto bail_out;
3689                         if (flags & ALLOW_MS2) {
3690                                 flags &= ~ALLOW_MS2;
3691                                 flags |= ALLOW_END_ONLY;
3692                                 continue;
3693                         }
3694                         goto bail_out;
3695 
3696                 case '0': case '1': case '2': case '3': case '4':
3697                 case '5': case '6': case '7': case '8': case '9':
3698 overflow:
3699                         (void) fprintf(stderr,
3700                             gettext("mkfs: value for %s overflowed\n"),
3701                             param);
3702                         while ((*cs != '\0') && (*cs != ','))
3703                                 cs++;
3704                         string = cs;
3705                         return (BIG);
3706 
3707                 default:
3708 bail_out:
3709                         (void) fprintf(stderr, gettext(
3710                             "mkfs: bad numeric arg for %s: \"%s\"\n"),
3711                             param, string);
3712                         while ((*cs != '\0') && (*cs != ','))
3713                                 cs++;
3714                         string = cs;
3715                         if (d_value != NO_DEFAULT) {
3716                                 (void) fprintf(stderr,
3717                                     gettext("mkfs: %s reset to default %lld\n"),
3718                                     param, d_value);
3719                                 return (d_value);
3720                         }
3721                         lockexit(2);
3722 
3723                 }
3724         } /* never gets here */
3725 }
3726 
3727 /* match ************************************************************** */
3728 /*                                                                      */
3729 /* Compare two text strings for equality                                */
3730 /*                                                                      */
3731 /* Arg:  s - pointer to string to match with a command arg              */
3732 /* Global arg:  string - pointer to command arg                         */
3733 /*                                                                      */
3734 /* Return:      1 if match, 0 if no match                               */
3735 /*              If match, also reset `string' to point to the text      */
3736 /*              that follows the matching text.                         */
3737 /*                                                                      */
3738 /* ******************************************************************** */
3739 
3740 static int
3741 match(char *s)
3742 {
3743         char *cs;
3744 
3745         cs = string;
3746         while (*cs++ == *s) {
3747                 if (*s++ == '\0') {
3748                         goto true;
3749                 }
3750         }
3751         if (*s != '\0') {
3752                 return (0);
3753         }
3754 
3755 true:
3756         cs--;
3757         string = cs;
3758         return (1);
3759 }
3760 
3761 /*
3762  * GROWFS ROUTINES
3763  */
3764 
3765 /* ARGSUSED */
3766 void
3767 lockexit(int exitstatus)
3768 {
3769         if (Pflag) {
3770                 /* the probe mode neither changes nor locks the filesystem */
3771                 exit(exitstatus);
3772         }
3773 
3774         /*
3775          * flush the dirty cylinder group
3776          */
3777         if (inlockexit == 0) {
3778                 inlockexit = 1;
3779                 flcg();
3780         }
3781 
3782         if (aio_inited) {
3783                 flush_writes();
3784         }
3785 
3786         /*
3787          * make sure the file system is unlocked before exiting
3788          */
3789         if ((inlockexit == 1) && (!isbad)) {
3790                 inlockexit = 2;
3791                 ulockfs();
3792                 /*
3793                  * if logging was enabled, then re-enable it
3794                  */
3795                 if (waslog) {
3796                         if (rl_log_control(fsys, _FIOLOGENABLE) != RL_SUCCESS) {
3797                                 (void) fprintf(stderr, gettext(
3798                                     "failed to re-enable logging\n"));
3799                         }
3800                 }
3801         } else if (grow) {
3802                 if (isbad) {
3803                         (void) fprintf(stderr, gettext(
3804                             "Filesystem is currently inconsistent.  It "
3805                             "must be repaired with fsck(1M)\nbefore being "
3806                             "used.  Use the following command to "
3807                             "do this:\n\n\tfsck %s\n\n"), fsys);
3808 
3809                         if (ismounted) {
3810                                 (void) fprintf(stderr, gettext(
3811                                     "You will be told that the filesystem "
3812                                     "is already mounted, and asked if you\n"
3813                                     "wish to continue.  Answer `yes' to "
3814                                     "this question.\n\n"));
3815                         }
3816 
3817                         (void) fprintf(stderr, gettext(
3818                             "One problem should be reported, that the summary "
3819                             "information is bad.\nYou will then be asked if it "
3820                             "should be salvaged.  Answer `yes' to\nthis "
3821                             "question.\n\n"));
3822                 }
3823 
3824                 if (ismounted) {
3825                         /*
3826                          * In theory, there's no way to get here without
3827                          * isbad also being set, but be robust in the
3828                          * face of future code changes.
3829                          */
3830                         (void) fprintf(stderr, gettext(
3831                             "The filesystem is currently mounted "
3832                             "read-only and write-locked.  "));
3833                         if (isbad) {
3834                                 (void) fprintf(stderr, gettext(
3835                                     "After\nrunning fsck, unlock the "
3836                                     "filesystem and "));
3837                         } else {
3838                                 (void) fprintf(stderr, gettext(
3839                                     "Unlock the filesystem\nand "));
3840                         }
3841 
3842                         (void) fprintf(stderr, gettext(
3843                             "re-enable writing with\nthe following "
3844                             "command:\n\n\tlockfs -u %s\n\n"), directory);
3845                 }
3846         }
3847 
3848         exit(exitstatus);
3849 }
3850 
3851 void
3852 randomgeneration()
3853 {
3854         int              i;
3855         struct dinode   *dp;
3856 
3857         /*
3858          * always perform fsirand(1) function... newfs will notice that
3859          * the inodes have been randomized and will not call fsirand itself
3860          */
3861         for (i = 0, dp = zino; i < sblock.fs_inopb; ++i, ++dp)
3862                 IRANDOMIZE(&dp->di_ic);
3863 }
3864 
3865 /*
3866  * Check the size of the summary information.
3867  * Fields in sblock are not changed in this function.
3868  *
3869  * For an 8K filesystem block, the maximum number of cylinder groups is 16384.
3870  *     MAXCSBUFS {32}  *   8K  {FS block size}
3871  *                         divided by (sizeof csum) {16}
3872  *
3873  * Note that MAXCSBUFS is not used in the kernel; as of Solaris 2.6 build 32,
3874  * this is the only place where it's referenced.
3875  */
3876 void
3877 checksummarysize()
3878 {
3879         diskaddr_t      dmax;
3880         diskaddr_t      dmin;
3881         int64_t cg0frags;
3882         int64_t cg0blocks;
3883         int64_t maxncg;
3884         int64_t maxfrags;
3885         uint64_t        fs_size;
3886         uint64_t maxfs_blocks; /* filesystem blocks for max filesystem size */
3887 
3888         /*
3889          * compute the maximum summary info size
3890          */
3891         dmin = cgdmin(&sblock, 0);
3892         dmax = cgbase(&sblock, 0) + sblock.fs_fpg;
3893         fs_size = (grow) ? grow_fs_size : sblock.fs_size;
3894         if (dmax > fs_size)
3895                 dmax = fs_size;
3896         cg0frags  = dmax - dmin;
3897         cg0blocks = cg0frags / sblock.fs_frag;
3898         cg0frags = cg0blocks * sblock.fs_frag;
3899         maxncg   = (longlong_t)cg0blocks *
3900             (longlong_t)(sblock.fs_bsize / sizeof (struct csum));
3901 
3902         maxfs_blocks = FS_MAX;
3903 
3904         if (maxncg > ((longlong_t)maxfs_blocks / (longlong_t)sblock.fs_fpg) + 1)
3905                 maxncg = ((longlong_t)maxfs_blocks /
3906                     (longlong_t)sblock.fs_fpg) + 1;
3907 
3908         maxfrags = maxncg * (longlong_t)sblock.fs_fpg;
3909 
3910         if (maxfrags > maxfs_blocks)
3911                 maxfrags = maxfs_blocks;
3912 
3913 
3914         /*
3915          * remember for later processing in extendsummaryinfo()
3916          */
3917         if (test)
3918                 grow_sifrag = dmin + (cg0blocks * sblock.fs_frag);
3919         if (testfrags == 0)
3920                 testfrags = cg0frags;
3921         if (testforce)
3922                 if (testfrags > cg0frags) {
3923                         (void) fprintf(stderr,
3924                             gettext("Too many test frags (%lld); "
3925                             "try %lld\n"), testfrags, cg0frags);
3926                         lockexit(32);
3927                 }
3928 
3929         /*
3930          * if summary info is too large (too many cg's) tell the user and exit
3931          */
3932         if ((longlong_t)sblock.fs_size > maxfrags) {
3933                 (void) fprintf(stderr, gettext(
3934                     "Too many cylinder groups with %llu sectors;\n    try "
3935                     "increasing cgsize, or decreasing fssize to %llu\n"),
3936                     fsbtodb(&sblock, (uint64_t)sblock.fs_size),
3937                     fsbtodb(&sblock, (uint64_t)maxfrags));
3938                 lockexit(32);
3939         }
3940 }
3941 
3942 /*
3943  * checksblock() has two uses:
3944  *      - One is to sanity test the superblock and is used when newfs(1M)
3945  *        is invoked with the "-N" option. If any discrepancy was found,
3946  *        just return whatever error was found and do not exit.
3947  *      - the other use of it is in places where you expect the superblock
3948  *        to be sane, and if it isn't, then we exit.
3949  * Which of the above two actions to take is indicated with the second argument.
3950  */
3951 
3952 int
3953 checksblock(struct fs sb, int proceed)
3954 {
3955         int err = 0;
3956         char *errmsg;
3957 
3958         if ((sb.fs_magic != FS_MAGIC) && (sb.fs_magic != MTB_UFS_MAGIC)) {
3959                 err = 1;
3960                 errmsg = gettext("Bad superblock; magic number wrong\n");
3961         } else if ((sb.fs_magic == FS_MAGIC &&
3962             (sb.fs_version != UFS_EFISTYLE4NONEFI_VERSION_2 &&
3963             sb.fs_version != UFS_VERSION_MIN)) ||
3964             (sb.fs_magic == MTB_UFS_MAGIC &&
3965             (sb.fs_version > MTB_UFS_VERSION_1 ||
3966             sb.fs_version < MTB_UFS_VERSION_MIN))) {
3967                 err = 2;
3968                 errmsg = gettext("Unrecognized version of UFS\n");
3969         } else if (sb.fs_ncg < 1) {
3970                 err = 3;
3971                 errmsg = gettext("Bad superblock; ncg out of range\n");
3972         } else if (sb.fs_cpg < 1) {
3973                 err = 4;
3974                 errmsg = gettext("Bad superblock; cpg out of range\n");
3975         } else if (sb.fs_ncg * sb.fs_cpg < sb.fs_ncyl ||
3976             (sb.fs_ncg - 1) * sb.fs_cpg >= sb.fs_ncyl) {
3977                 err = 5;
3978                 errmsg = gettext("Bad superblock; ncyl out of range\n");
3979         } else if (sb.fs_sbsize <= 0 || sb.fs_sbsize > sb.fs_bsize) {
3980                 err = 6;
3981                 errmsg = gettext("Bad superblock; superblock size out of "
3982                     "range\n");
3983         }
3984 
3985         if (proceed) {
3986                 if (err) dprintf(("%s", errmsg));
3987                 return (err);
3988         }
3989 
3990         if (err) {
3991                 fprintf(stderr, "%s", errmsg);
3992                 lockexit(32);
3993         }
3994         return (32);
3995 }
3996 
3997 /*
3998  * Roll the embedded log, if any, and set up the global variables
3999  * islog and islogok.
4000  */
4001 static void
4002 logsetup(char *devstr)
4003 {
4004         void            *buf, *ud_buf;
4005         extent_block_t  *ebp;
4006         ml_unit_t       *ul;
4007         ml_odunit_t     *ud;
4008 
4009         /*
4010          * Does the superblock indicate that we are supposed to have a log ?
4011          */
4012         if (sblock.fs_logbno == 0) {
4013                 /*
4014                  * No log present, nothing to do.
4015                  */
4016                 islog = 0;
4017                 islogok = 0;
4018                 return;
4019         } else {
4020                 /*
4021                  * There's a log in a yet unknown state, attempt to roll it.
4022                  */
4023                 islogok = 0;
4024 
4025                 /*
4026                  * We failed to roll the log, bail out.
4027                  */
4028                 if (rl_roll_log(devstr) != RL_SUCCESS)
4029                         return;
4030 
4031                 islog = 1;
4032 
4033                 /* log is not okay; check the fs */
4034                 if ((FSOKAY != (sblock.fs_state + sblock.fs_time)) ||
4035                     (sblock.fs_clean != FSLOG))
4036                         return;
4037 
4038                 /* get the log allocation block */
4039                 buf = (void *)malloc(DEV_BSIZE);
4040                 if (buf == (void *) NULL)
4041                         return;
4042 
4043                 ud_buf = (void *)malloc(DEV_BSIZE);
4044                 if (ud_buf == (void *) NULL) {
4045                         free(buf);
4046                         return;
4047                 }
4048 
4049                 rdfs((diskaddr_t)logbtodb(&sblock, sblock.fs_logbno),
4050                     DEV_BSIZE, buf);
4051                 ebp = (extent_block_t *)buf;
4052 
4053                 /* log allocation block is not okay; check the fs */
4054                 if (ebp->type != LUFS_EXTENTS) {
4055                         free(buf);
4056                         free(ud_buf);
4057                         return;
4058                 }
4059 
4060                 /* get the log state block(s) */
4061                 rdfs((diskaddr_t)logbtodb(&sblock, ebp->extents[0].pbno),
4062                     DEV_BSIZE, ud_buf);
4063                 ud = (ml_odunit_t *)ud_buf;
4064                 ul = (ml_unit_t *)malloc(sizeof (*ul));
4065                 ul->un_ondisk = *ud;
4066 
4067                 /* log state is okay */
4068                 if ((ul->un_chksum == ul->un_head_ident + ul->un_tail_ident) &&
4069                     (ul->un_version == LUFS_VERSION_LATEST) &&
4070                     (ul->un_badlog == 0))
4071                         islogok = 1;
4072                 free(ud_buf);
4073                 free(buf);
4074                 free(ul);
4075         }
4076 }
4077 
4078 void
4079 growinit(char *devstr)
4080 {
4081         int     i;
4082         char    buf[DEV_BSIZE];
4083 
4084         /*
4085          * Read and verify the superblock
4086          */
4087         rdfs((diskaddr_t)(SBOFF / sectorsize), (int)sbsize, (char *)&sblock);
4088         (void) checksblock(sblock, 0);
4089         if (sblock.fs_postblformat != FS_DYNAMICPOSTBLFMT) {
4090                 (void) fprintf(stderr,
4091                     gettext("old file system format; can't growfs\n"));
4092                 lockexit(32);
4093         }
4094 
4095         /*
4096          * can't shrink a file system
4097          */
4098         grow_fssize = fsbtodb(&sblock, (uint64_t)sblock.fs_size);
4099         if (fssize_db < grow_fssize) {
4100                 (void) fprintf(stderr,
4101                     gettext("%lld sectors < current size of %lld sectors\n"),
4102                     fssize_db, grow_fssize);
4103                 lockexit(32);
4104         }
4105 
4106         /*
4107          * can't grow a system to over a terabyte unless it was set up
4108          * as an MTB UFS file system.
4109          */
4110         if (mtb == 'y' && sblock.fs_magic != MTB_UFS_MAGIC) {
4111                 if (fssize_db >= SECTORS_PER_TERABYTE) {
4112                         (void) fprintf(stderr, gettext(
4113 "File system was not set up with the multi-terabyte format.\n"));
4114                         (void) fprintf(stderr, gettext(
4115 "Its size cannot be increased to a terabyte or more.\n"));
4116                 } else {
4117                         (void) fprintf(stderr, gettext(
4118 "Cannot convert file system to multi-terabyte format.\n"));
4119                 }
4120                 lockexit(32);
4121         }
4122 
4123         logsetup(devstr);
4124 
4125         /*
4126          * can't growfs when logging device has errors
4127          */
4128         if ((islog && !islogok) ||
4129             ((FSOKAY == (sblock.fs_state + sblock.fs_time)) &&
4130             (sblock.fs_clean == FSLOG && !islog))) {
4131                 (void) fprintf(stderr,
4132                     gettext("logging device has errors; can't growfs\n"));
4133                 lockexit(32);
4134         }
4135 
4136         /*
4137          * disable ufs logging for growing
4138          */
4139         if (islog) {
4140                 if (rl_log_control(devstr, _FIOLOGDISABLE) != RL_SUCCESS) {
4141                         (void) fprintf(stderr, gettext(
4142                             "failed to disable logging\n"));
4143                         lockexit(32);
4144                 }
4145                 islog = 0;
4146                 waslog = 1;
4147         }
4148 
4149         /*
4150          * if mounted write lock the file system to be grown
4151          */
4152         if (ismounted)
4153                 wlockfs();
4154 
4155         /*
4156          * refresh dynamic superblock state - disabling logging will have
4157          * changed the amount of free space available in the file system
4158          */
4159         rdfs((diskaddr_t)(SBOFF / sectorsize), sbsize, (char *)&sblock);
4160 
4161         /*
4162          * make sure device is big enough
4163          */
4164         rdfs((diskaddr_t)fssize_db - 1, DEV_BSIZE, buf);
4165         wtfs((diskaddr_t)fssize_db - 1, DEV_BSIZE, buf);
4166 
4167         /*
4168          * read current summary information
4169          */
4170         grow_fscs = read_summaryinfo(&sblock);
4171 
4172         /*
4173          * save some current size related fields from the superblock
4174          * These are used in extendsummaryinfo()
4175          */
4176         grow_fs_size    = sblock.fs_size;
4177         grow_fs_ncg     = sblock.fs_ncg;
4178         grow_fs_csaddr  = (diskaddr_t)sblock.fs_csaddr;
4179         grow_fs_cssize  = sblock.fs_cssize;
4180 
4181         /*
4182          * save and reset the clean flag
4183          */
4184         if (FSOKAY == (sblock.fs_state + sblock.fs_time))
4185                 grow_fs_clean = sblock.fs_clean;
4186         else
4187                 grow_fs_clean = FSBAD;
4188         sblock.fs_clean = FSBAD;
4189         sblock.fs_state = FSOKAY - sblock.fs_time;
4190         isbad = 1;
4191         wtfs((diskaddr_t)(SBOFF / sectorsize), sbsize, (char *)&sblock);
4192 }
4193 
4194 void
4195 checkdev(char *rdev, char *bdev)
4196 {
4197         struct stat64   statarea;
4198 
4199         if (stat64(bdev, &statarea) < 0) {
4200                 (void) fprintf(stderr, gettext("can't check mount point; "));
4201                 (void) fprintf(stderr, gettext("can't stat %s\n"), bdev);
4202                 lockexit(32);
4203         }
4204         if ((statarea.st_mode & S_IFMT) != S_IFBLK) {
4205                 (void) fprintf(stderr, gettext(
4206                     "can't check mount point; %s is not a block device\n"),
4207                     bdev);
4208                 lockexit(32);
4209         }
4210         if (stat64(rdev, &statarea) < 0) {
4211                 (void) fprintf(stderr, gettext("can't stat %s\n"), rdev);
4212                 lockexit(32);
4213         }
4214         if ((statarea.st_mode & S_IFMT) != S_IFCHR) {
4215                 (void) fprintf(stderr,
4216                     gettext("%s is not a character device\n"), rdev);
4217                 lockexit(32);
4218         }
4219 }
4220 
4221 void
4222 checkmount(struct mnttab *mntp, char *bdevname)
4223 {
4224         struct stat64   statdir;
4225         struct stat64   statdev;
4226 
4227         if (strcmp(bdevname, mntp->mnt_special) == 0) {
4228                 if (stat64(mntp->mnt_mountp, &statdir) == -1) {
4229                         (void) fprintf(stderr, gettext("can't stat %s\n"),
4230                             mntp->mnt_mountp);
4231                         lockexit(32);
4232                 }
4233                 if (stat64(mntp->mnt_special, &statdev) == -1) {
4234                         (void) fprintf(stderr, gettext("can't stat %s\n"),
4235                             mntp->mnt_special);
4236                         lockexit(32);
4237                 }
4238                 if (statdir.st_dev != statdev.st_rdev) {
4239                         (void) fprintf(stderr, gettext(
4240                             "%s is not mounted on %s; mnttab(4) wrong\n"),
4241                             mntp->mnt_special, mntp->mnt_mountp);
4242                         lockexit(32);
4243                 }
4244                 ismounted = 1;
4245                 if (directory) {
4246                         if (strcmp(mntp->mnt_mountp, directory) != 0) {
4247                                 (void) fprintf(stderr,
4248                                     gettext("%s is mounted on %s, not %s\n"),
4249                                     bdevname, mntp->mnt_mountp, directory);
4250                                 lockexit(32);
4251                         }
4252                 } else {
4253                         if (grow)
4254                                 (void) fprintf(stderr, gettext(
4255                                     "%s is mounted on %s; can't growfs\n"),
4256                                     bdevname, mntp->mnt_mountp);
4257                         else
4258                                 (void) fprintf(stderr,
4259                                     gettext("%s is mounted, can't mkfs\n"),
4260                                     bdevname);
4261                         lockexit(32);
4262                 }
4263         }
4264 }
4265 
4266 struct dinode   *dibuf  = 0;
4267 diskaddr_t      difrag  = 0;
4268 
4269 struct dinode *
4270 gdinode(ino_t ino)
4271 {
4272         /*
4273          * read the block of inodes containing inode number ino
4274          */
4275         if (dibuf == 0)
4276                 dibuf = (struct dinode *)malloc((unsigned)sblock.fs_bsize);
4277         if (itod(&sblock, ino) != difrag) {
4278                 difrag = itod(&sblock, ino);
4279                 rdfs(fsbtodb(&sblock, (uint64_t)difrag), (int)sblock.fs_bsize,
4280                     (char *)dibuf);
4281         }
4282         return (dibuf + (ino % INOPB(&sblock)));
4283 }
4284 
4285 /*
4286  * structure that manages the frags we need for extended summary info
4287  *      These frags can be:
4288  *              free
4289  *              data  block
4290  *              alloc block
4291  */
4292 struct csfrag {
4293         struct csfrag   *next;          /* next entry */
4294         daddr32_t        ofrag;         /* old frag */
4295         daddr32_t        nfrag;         /* new frag */
4296         long             cylno;         /* cylno of nfrag */
4297         long             frags;         /* number of frags */
4298         long             size;          /* size in bytes */
4299         ino_t            ino;           /* inode number */
4300         long             fixed;         /* Boolean - Already fixed? */
4301 };
4302 struct csfrag   *csfrag;                /* state unknown */
4303 struct csfrag   *csfragino;             /* frags belonging to an inode */
4304 struct csfrag   *csfragfree;            /* frags that are free */
4305 
4306 daddr32_t maxcsfrag     = 0;            /* maximum in range */
4307 daddr32_t mincsfrag     = 0x7fffffff;   /* minimum in range */
4308 
4309 int
4310 csfraginrange(daddr32_t frag)
4311 {
4312         return ((frag >= mincsfrag) && (frag <= maxcsfrag));
4313 }
4314 
4315 struct csfrag *
4316 findcsfrag(daddr32_t frag, struct csfrag **cfap)
4317 {
4318         struct csfrag   *cfp;
4319 
4320         if (!csfraginrange(frag))
4321                 return (NULL);
4322 
4323         for (cfp = *cfap; cfp; cfp = cfp->next)
4324                 if (cfp->ofrag == frag)
4325                         return (cfp);
4326         return (NULL);
4327 }
4328 
4329 void
4330 checkindirect(ino_t ino, daddr32_t *fragsp, daddr32_t frag, int level)
4331 {
4332         int                     i;
4333         int                     ne      = sblock.fs_bsize / sizeof (daddr32_t);
4334         daddr32_t                       fsb[MAXBSIZE / sizeof (daddr32_t)];
4335 
4336         if (frag == 0)
4337                 return;
4338 
4339         rdfs(fsbtodb(&sblock, frag), (int)sblock.fs_bsize,
4340             (char *)fsb);
4341 
4342         checkdirect(ino, fragsp, fsb, sblock.fs_bsize / sizeof (daddr32_t));
4343 
4344         if (level)
4345                 for (i = 0; i < ne && *fragsp; ++i)
4346                         checkindirect(ino, fragsp, fsb[i], level-1);
4347 }
4348 
4349 void
4350 addcsfrag(ino_t ino, daddr32_t frag, struct csfrag **cfap)
4351 {
4352         struct csfrag   *cfp, *curr, *prev;
4353 
4354         /*
4355          * establish a range for faster checking in csfraginrange()
4356          */
4357         if (frag > maxcsfrag)
4358                 maxcsfrag = frag;
4359         if (frag < mincsfrag)
4360                 mincsfrag = frag;
4361 
4362         /*
4363          * if this frag belongs to an inode and is not the start of a block
4364          *      then see if it is part of a frag range for this inode
4365          */
4366         if (ino && (frag % sblock.fs_frag))
4367                 for (cfp = *cfap; cfp; cfp = cfp->next) {
4368                         if (ino != cfp->ino)
4369                                 continue;
4370                         if (frag != cfp->ofrag + cfp->frags)
4371                                 continue;
4372                         cfp->frags++;
4373                         cfp->size += sblock.fs_fsize;
4374                         return;
4375                 }
4376         /*
4377          * allocate a csfrag entry and insert it in an increasing order into the
4378          * specified list
4379          */
4380         cfp = (struct csfrag *)calloc(1, sizeof (struct csfrag));
4381         cfp->ino     = ino;
4382         cfp->ofrag   = frag;
4383         cfp->frags   = 1;
4384         cfp->size    = sblock.fs_fsize;
4385         for (prev = NULL, curr = *cfap; curr != NULL;
4386             prev = curr, curr = curr->next) {
4387                 if (frag < curr->ofrag) {
4388                         cfp->next = curr;
4389                         if (prev)
4390                                 prev->next = cfp;    /* middle element */
4391                         else
4392                                 *cfap = cfp;            /* first element */
4393                         break;
4394                 }
4395                 if (curr->next == NULL) {
4396                         curr->next = cfp;            /* last element */
4397                         break;
4398                 }
4399         }
4400         if (*cfap == NULL)      /* will happen only once */
4401                 *cfap = cfp;
4402 }
4403 
4404 void
4405 delcsfrag(daddr32_t frag, struct csfrag **cfap)
4406 {
4407         struct csfrag   *cfp;
4408         struct csfrag   **cfpp;
4409 
4410         /*
4411          * free up entry whose beginning frag matches
4412          */
4413         for (cfpp = cfap; *cfpp; cfpp = &(*cfpp)->next) {
4414                 if (frag == (*cfpp)->ofrag) {
4415                         cfp = *cfpp;
4416                         *cfpp = (*cfpp)->next;
4417                         free((char *)cfp);
4418                         return;
4419                 }
4420         }
4421 }
4422 
4423 /*
4424  * See whether any of the direct blocks in the array pointed by "db" and of
4425  * length "ne" are within the range of frags needed to extend the cylinder
4426  * summary. If so, remove those frags from the "as-yet-unclassified" list
4427  * (csfrag) and add them to the "owned-by-inode" list (csfragino).
4428  * For each such frag found, decrement the frag count pointed to by fragsp.
4429  * "ino" is the inode that contains (either directly or indirectly) the frags
4430  * being checked.
4431  */
4432 void
4433 checkdirect(ino_t ino, daddr32_t *fragsp, daddr32_t *db, int ne)
4434 {
4435         int      i;
4436         int      j;
4437         int      found;
4438         diskaddr_t       frag;
4439 
4440         /*
4441          * scan for allocation within the new summary info range
4442          */
4443         for (i = 0; i < ne && *fragsp; ++i) {
4444                 if ((frag = *db++) != 0) {
4445                         found = 0;
4446                         for (j = 0; j < sblock.fs_frag && *fragsp; ++j) {
4447                                 if (found || (found = csfraginrange(frag))) {
4448                                         addcsfrag(ino, frag, &csfragino);
4449                                         delcsfrag(frag, &csfrag);
4450                                 }
4451                                 ++frag;
4452                                 --(*fragsp);
4453                         }
4454                 }
4455         }
4456 }
4457 
4458 void
4459 findcsfragino()
4460 {
4461         int              i;
4462         int              j;
4463         daddr32_t                frags;
4464         struct dinode   *dp;
4465 
4466         /*
4467          * scan all old inodes looking for allocations in the new
4468          * summary info range.  Move the affected frag from the
4469          * generic csfrag list onto the `owned-by-inode' list csfragino.
4470          */
4471         for (i = UFSROOTINO; i < grow_fs_ncg*sblock.fs_ipg && csfrag; ++i) {
4472                 dp = gdinode((ino_t)i);
4473                 switch (dp->di_mode & IFMT) {
4474                         case IFSHAD     :
4475                         case IFLNK      :
4476                         case IFDIR      :
4477                         case IFREG      : break;
4478                         default         : continue;
4479                 }
4480 
4481                 frags   = dbtofsb(&sblock, dp->di_blocks);
4482 
4483                 checkdirect((ino_t)i, &frags, &dp->di_db[0], NDADDR+NIADDR);
4484                 for (j = 0; j < NIADDR && frags; ++j) {
4485                         /* Negate the block if its an fallocate'd block */
4486                         if (dp->di_ib[j] < 0 && dp->di_ib[j] != UFS_HOLE)
4487                                 checkindirect((ino_t)i, &frags,
4488                                     -(dp->di_ib[j]), j);
4489                         else
4490                                 checkindirect((ino_t)i, &frags,
4491                                     dp->di_ib[j], j);
4492                 }
4493         }
4494 }
4495 
4496 void
4497 fixindirect(daddr32_t frag, int level)
4498 {
4499         int                      i;
4500         int                      ne     = sblock.fs_bsize / sizeof (daddr32_t);
4501         daddr32_t                       fsb[MAXBSIZE / sizeof (daddr32_t)];
4502 
4503         if (frag == 0)
4504                 return;
4505 
4506         rdfs(fsbtodb(&sblock, (uint64_t)frag), (int)sblock.fs_bsize,
4507             (char *)fsb);
4508 
4509         fixdirect((caddr_t)fsb, frag, fsb, ne);
4510 
4511         if (level)
4512                 for (i = 0; i < ne; ++i)
4513                         fixindirect(fsb[i], level-1);
4514 }
4515 
4516 void
4517 fixdirect(caddr_t bp, daddr32_t frag, daddr32_t *db, int ne)
4518 {
4519         int      i;
4520         struct csfrag   *cfp;
4521 
4522         for (i = 0; i < ne; ++i, ++db) {
4523                 if (*db == 0)
4524                         continue;
4525                 if ((cfp = findcsfrag(*db, &csfragino)) == NULL)
4526                         continue;
4527                 *db = cfp->nfrag;
4528                 cfp->fixed = 1;
4529                 wtfs(fsbtodb(&sblock, (uint64_t)frag), (int)sblock.fs_bsize,
4530                     bp);
4531         }
4532 }
4533 
4534 void
4535 fixcsfragino()
4536 {
4537         int              i;
4538         struct dinode   *dp;
4539         struct csfrag   *cfp;
4540 
4541         for (cfp = csfragino; cfp; cfp = cfp->next) {
4542                 if (cfp->fixed)
4543                         continue;
4544                 dp = gdinode((ino_t)cfp->ino);
4545                 fixdirect((caddr_t)dibuf, difrag, dp->di_db, NDADDR+NIADDR);
4546                 for (i = 0; i < NIADDR; ++i)
4547                         fixindirect(dp->di_ib[i], i);
4548         }
4549 }
4550 
4551 /*
4552  * Read the cylinders summary information specified by settings in the
4553  * passed 'fs' structure into a new allocated array of csum structures.
4554  * The caller is responsible for freeing the returned array.
4555  * Return a pointer to an array of csum structures.
4556  */
4557 static struct csum *
4558 read_summaryinfo(struct fs *fsp)
4559 {
4560         struct csum     *csp;
4561         int             i;
4562 
4563         if ((csp = malloc((size_t)fsp->fs_cssize)) == NULL) {
4564                 (void) fprintf(stderr, gettext("cannot create csum list,"
4565                     " not enough memory\n"));
4566                 exit(32);
4567         }
4568 
4569         for (i = 0; i < fsp->fs_cssize; i += fsp->fs_bsize) {
4570                 rdfs(fsbtodb(fsp,
4571                     (uint64_t)(fsp->fs_csaddr + numfrags(fsp, i))),
4572                     (int)(fsp->fs_cssize - i < fsp->fs_bsize ?
4573                     fsp->fs_cssize - i : fsp->fs_bsize), ((caddr_t)csp) + i);
4574         }
4575 
4576         return (csp);
4577 }
4578 
4579 /*
4580  * Check the allocation of fragments that are to be made part of a csum block.
4581  * A fragment is allocated if it is either in the csfragfree list or, it is
4582  * in the csfragino list and has new frags associated with it.
4583  * Return the number of allocated fragments.
4584  */
4585 int64_t
4586 checkfragallocated(daddr32_t frag)
4587 {
4588         struct  csfrag  *cfp;
4589         /*
4590          * Since the lists are sorted we can break the search if the asked
4591          * frag is smaller then the one in the list.
4592          */
4593         for (cfp = csfragfree; cfp != NULL && frag >= cfp->ofrag;
4594             cfp = cfp->next) {
4595                 if (frag == cfp->ofrag)
4596                         return (1);
4597         }
4598         for (cfp = csfragino; cfp != NULL && frag >= cfp->ofrag;
4599             cfp = cfp->next) {
4600                 if (frag == cfp->ofrag && cfp->nfrag != 0)
4601                         return (cfp->frags);
4602         }
4603 
4604         return (0);
4605 }
4606 
4607 /*
4608  * Figure out how much the filesystem can be grown. The limiting factor is
4609  * the available free space needed to extend the cg summary info block.
4610  * The free space is determined in three steps:
4611  * - Try to extend the cg summary block to the required size.
4612  * - Find free blocks in last cg.
4613  * - Find free space in the last already allocated fragment of the summary info
4614  *   block, and use it for additional csum structures.
4615  * Return the maximum size of the new filesystem or 0 if it can't be grown.
4616  * Please note that this function leaves the global list pointers csfrag,
4617  * csfragfree, and csfragino initialized, and the caller is responsible for
4618  * freeing the lists.
4619  */
4620 diskaddr_t
4621 probe_summaryinfo()
4622 {
4623         /* fragments by which the csum block can be extended. */
4624         int64_t         growth_csum_frags = 0;
4625         /* fragments by which the filesystem can be extended. */
4626         int64_t         growth_fs_frags = 0;
4627         int64_t         new_fs_cssize;  /* size of csum blk in the new FS */
4628         int64_t         new_fs_ncg;     /* number of cg in the new FS */
4629         int64_t         spare_csum;
4630         daddr32_t       oldfrag_daddr;
4631         daddr32_t       newfrag_daddr;
4632         daddr32_t       daddr;
4633         int             i;
4634 
4635         /*
4636          * read and verify the superblock
4637          */
4638         rdfs((diskaddr_t)(SBOFF / sectorsize), (int)sbsize, (char *)&sblock);
4639         (void) checksblock(sblock, 0);
4640 
4641         /*
4642          * check how much we can extend the cg summary info block
4643          */
4644 
4645         /*
4646          * read current summary information
4647          */
4648         fscs = read_summaryinfo(&sblock);
4649 
4650         /*
4651          * build list of frags needed for cg summary info block extension
4652          */
4653         oldfrag_daddr = howmany(sblock.fs_cssize, sblock.fs_fsize) +
4654             sblock.fs_csaddr;
4655         new_fs_ncg = howmany(dbtofsb(&sblock, fssize_db), sblock.fs_fpg);
4656         new_fs_cssize = fragroundup(&sblock, new_fs_ncg * sizeof (struct csum));
4657         newfrag_daddr = howmany(new_fs_cssize, sblock.fs_fsize) +
4658             sblock.fs_csaddr;
4659         /*
4660          * add all of the frags that are required to grow the cyl summary to the
4661          * csfrag list, which is the generic/unknown list, since at this point
4662          * we don't yet know the state of those frags.
4663          */
4664         for (daddr = oldfrag_daddr; daddr < newfrag_daddr; daddr++)
4665                 addcsfrag((ino_t)0, daddr, &csfrag);
4666 
4667         /*
4668          * filter free fragments and allocate them. Note that the free frags
4669          * must be allocated first otherwise they could be grabbed by
4670          * alloccsfragino() for data frags.
4671          */
4672         findcsfragfree();
4673         alloccsfragfree();
4674 
4675         /*
4676          * filter fragments owned by inodes and allocate them
4677          */
4678         grow_fs_ncg = sblock.fs_ncg; /* findcsfragino() needs this glob. var. */
4679         findcsfragino();
4680         alloccsfragino();
4681 
4682         if (notenoughspace()) {
4683                 /*
4684                  * check how many consecutive fragments could be allocated
4685                  * in both lists.
4686                  */
4687                 int64_t tmp_frags;
4688                 for (daddr = oldfrag_daddr; daddr < newfrag_daddr;
4689                     daddr += tmp_frags) {
4690                         if ((tmp_frags = checkfragallocated(daddr)) > 0)
4691                                 growth_csum_frags += tmp_frags;
4692                         else
4693                                 break;
4694                 }
4695         } else {
4696                 /*
4697                  * We have all we need for the new desired size,
4698                  * so clean up and report back.
4699                  */
4700                 return (fssize_db);
4701         }
4702 
4703         /*
4704          * given the number of fragments by which the csum block can be grown
4705          * compute by how many new fragments the FS can be increased.
4706          * It is the number of csum instances per fragment multiplied by
4707          * `growth_csum_frags' and the number of fragments per cylinder group.
4708          */
4709         growth_fs_frags = howmany(sblock.fs_fsize, sizeof (struct csum)) *
4710             growth_csum_frags * sblock.fs_fpg;
4711 
4712         /*
4713          * compute free fragments in the last cylinder group
4714          */
4715         rdcg(sblock.fs_ncg - 1);
4716         growth_fs_frags += sblock.fs_fpg - acg.cg_ndblk;
4717 
4718         /*
4719          * compute how many csum instances are unused in the old csum block.
4720          * For each unused csum instance the FS can be grown by one cylinder
4721          * group without extending the csum block.
4722          */
4723         spare_csum = howmany(sblock.fs_cssize, sizeof (struct csum)) -
4724             sblock.fs_ncg;
4725         if (spare_csum > 0)
4726                 growth_fs_frags += spare_csum * sblock.fs_fpg;
4727 
4728         /*
4729          * recalculate the new filesystem size in sectors, shorten it by
4730          * the requested size `fssize_db' if necessary.
4731          */
4732         if (growth_fs_frags > 0) {
4733                 diskaddr_t sect;
4734                 sect = (sblock.fs_size + growth_fs_frags) * sblock.fs_nspf;
4735                 return ((sect > fssize_db) ? fssize_db : sect);
4736         }
4737 
4738         return (0);
4739 }
4740 
4741 void
4742 extendsummaryinfo()
4743 {
4744         int64_t         i;
4745         int             localtest       = test;
4746         int64_t         frags;
4747         daddr32_t               oldfrag;
4748         daddr32_t               newfrag;
4749 
4750         /*
4751          * if no-write (-N), don't bother
4752          */
4753         if (Nflag)
4754                 return;
4755 
4756 again:
4757         flcg();
4758         /*
4759          * summary info did not change size -- do nothing unless in test mode
4760          */
4761         if (grow_fs_cssize == sblock.fs_cssize)
4762                 if (!localtest)
4763                         return;
4764 
4765         /*
4766          * build list of frags needed for additional summary information
4767          */
4768         oldfrag = howmany(grow_fs_cssize, sblock.fs_fsize) + grow_fs_csaddr;
4769         newfrag = howmany(sblock.fs_cssize, sblock.fs_fsize) + grow_fs_csaddr;
4770         /*
4771          * add all of the frags that are required to grow the cyl summary to the
4772          * csfrag list, which is the generic/unknown list, since at this point
4773          * we don't yet know the state of those frags.
4774          */
4775         for (i = oldfrag, frags = 0; i < newfrag; ++i, ++frags)
4776                 addcsfrag((ino_t)0, (diskaddr_t)i, &csfrag);
4777         /*
4778          * reduce the number of data blocks in the file system (fs_dsize) by
4779          * the number of frags that need to be added to the cyl summary
4780          */
4781         sblock.fs_dsize -= (newfrag - oldfrag);
4782 
4783         /*
4784          * In test mode, we move more data than necessary from
4785          * cylinder group 0.  The lookup/allocate/move code can be
4786          * better stressed without having to create HUGE file systems.
4787          */
4788         if (localtest)
4789                 for (i = newfrag; i < grow_sifrag; ++i) {
4790                         if (frags >= testfrags)
4791                                 break;
4792                         frags++;
4793                         addcsfrag((ino_t)0, (diskaddr_t)i, &csfrag);
4794                 }
4795 
4796         /*
4797          * move frags to free or inode lists, depending on owner
4798          */
4799         findcsfragfree();
4800         findcsfragino();
4801 
4802         /*
4803          * if not all frags can be located, file system must be inconsistent
4804          */
4805         if (csfrag) {
4806                 isbad = 1;      /* should already be set, but make sure */
4807                 lockexit(32);
4808         }
4809 
4810         /*
4811          * allocate the free frags. Note that the free frags must be allocated
4812          * first otherwise they could be grabbed by alloccsfragino() for data
4813          * frags.
4814          */
4815         alloccsfragfree();
4816         /*
4817          * allocate extra space for inode frags
4818          */
4819         alloccsfragino();
4820 
4821         /*
4822          * not enough space
4823          */
4824         if (notenoughspace()) {
4825                 unalloccsfragfree();
4826                 unalloccsfragino();
4827                 if (localtest && !testforce) {
4828                         localtest = 0;
4829                         goto again;
4830                 }
4831                 (void) fprintf(stderr, gettext("Not enough free space\n"));
4832                 lockexit(NOTENOUGHSPACE);
4833         }
4834 
4835         /*
4836          * copy the data from old frags to new frags
4837          */
4838         copycsfragino();
4839 
4840         /*
4841          * fix the inodes to point to the new frags
4842          */
4843         fixcsfragino();
4844 
4845         /*
4846          * We may have moved more frags than we needed.  Free them.
4847          */
4848         rdcg((long)0);
4849         for (i = newfrag; i <= maxcsfrag; ++i)
4850                 setbit(cg_blksfree(&acg), i-cgbase(&sblock, 0));
4851         wtcg();
4852 
4853         flcg();
4854 }
4855 
4856 /*
4857  * Check if all fragments in the `csfragino' list were reallocated.
4858  */
4859 int
4860 notenoughspace()
4861 {
4862         struct csfrag   *cfp;
4863 
4864         /*
4865          * If any element in the csfragino array has a "new frag location"
4866          * of 0, the allocfrags() function was unsuccessful in allocating
4867          * space for moving the frag represented by this array element.
4868          */
4869         for (cfp = csfragino; cfp; cfp = cfp->next)
4870                 if (cfp->nfrag == 0)
4871                         return (1);
4872         return (0);
4873 }
4874 
4875 void
4876 unalloccsfragino()
4877 {
4878         struct csfrag   *cfp;
4879 
4880         while ((cfp = csfragino) != NULL) {
4881                 if (cfp->nfrag)
4882                         freefrags(cfp->nfrag, cfp->frags, cfp->cylno);
4883                 delcsfrag(cfp->ofrag, &csfragino);
4884         }
4885 }
4886 
4887 void
4888 unalloccsfragfree()
4889 {
4890         struct csfrag   *cfp;
4891 
4892         while ((cfp = csfragfree) != NULL) {
4893                 freefrags(cfp->ofrag, cfp->frags, cfp->cylno);
4894                 delcsfrag(cfp->ofrag, &csfragfree);
4895         }
4896 }
4897 
4898 /*
4899  * For each frag in the "as-yet-unclassified" list (csfrag), see if
4900  * it's free (i.e., its bit is set in the free frag bit map).  If so,
4901  * move it from the "as-yet-unclassified" list to the csfragfree list.
4902  */
4903 void
4904 findcsfragfree()
4905 {
4906         struct csfrag   *cfp;
4907         struct csfrag   *cfpnext;
4908 
4909         /*
4910          * move free frags onto the free-frag list
4911          */
4912         rdcg((long)0);
4913         for (cfp = csfrag; cfp; cfp = cfpnext) {
4914                 cfpnext = cfp->next;
4915                 if (isset(cg_blksfree(&acg), cfp->ofrag - cgbase(&sblock, 0))) {
4916                         addcsfrag(cfp->ino, cfp->ofrag, &csfragfree);
4917                         delcsfrag(cfp->ofrag, &csfrag);
4918                 }
4919         }
4920 }
4921 
4922 void
4923 copycsfragino()
4924 {
4925         struct csfrag   *cfp;
4926         char            buf[MAXBSIZE];
4927 
4928         /*
4929          * copy data from old frags to newly allocated frags
4930          */
4931         for (cfp = csfragino; cfp; cfp = cfp->next) {
4932                 rdfs(fsbtodb(&sblock, (uint64_t)cfp->ofrag), (int)cfp->size,
4933                     buf);
4934                 wtfs(fsbtodb(&sblock, (uint64_t)cfp->nfrag), (int)cfp->size,
4935                     buf);
4936         }
4937 }
4938 
4939 long    curcylno        = -1;
4940 int     cylnodirty      = 0;
4941 
4942 void
4943 rdcg(long cylno)
4944 {
4945         if (cylno != curcylno) {
4946                 flcg();
4947                 curcylno = cylno;
4948                 rdfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, curcylno)),
4949                     (int)sblock.fs_cgsize, (char *)&acg);
4950         }
4951 }
4952 
4953 void
4954 flcg()
4955 {
4956         if (cylnodirty) {
4957                 if (debug && Pflag) {
4958                         (void) fprintf(stderr,
4959                             "Assert: cylnodirty set in probe mode\n");
4960                         return;
4961                 }
4962                 resetallocinfo();
4963                 wtfs(fsbtodb(&sblock, (uint64_t)cgtod(&sblock, curcylno)),
4964                     (int)sblock.fs_cgsize, (char *)&acg);
4965                 cylnodirty = 0;
4966         }
4967         curcylno = -1;
4968 }
4969 
4970 void
4971 wtcg()
4972 {
4973         if (!Pflag) {
4974                 /* probe mode should never write to disk */
4975                 cylnodirty = 1;
4976         }
4977 }
4978 
4979 void
4980 allocfrags(long frags, daddr32_t *fragp, long *cylnop)
4981 {
4982         int      i;
4983         int      j;
4984         long     bits;
4985         long     bit;
4986 
4987         /*
4988          * Allocate a free-frag range in an old cylinder group
4989          */
4990         for (i = 0, *fragp = 0; i < grow_fs_ncg; ++i) {
4991                 if (((fscs+i)->cs_nffree < frags) && ((fscs+i)->cs_nbfree == 0))
4992                         continue;
4993                 rdcg((long)i);
4994                 bit = bits = 0;
4995                 while (findfreerange(&bit, &bits)) {
4996                         if (frags <= bits)  {
4997                                 for (j = 0; j < frags; ++j)
4998                                         clrbit(cg_blksfree(&acg), bit+j);
4999                                 wtcg();
5000                                 *cylnop = i;
5001                                 *fragp  = bit + cgbase(&sblock, i);
5002                                 return;
5003                         }
5004                         bit += bits;
5005                 }
5006         }
5007 }
5008 
5009 /*
5010  * Allocate space for frags that need to be moved in order to free up space for
5011  * expanding the cylinder summary info.
5012  * For each frag that needs to be moved (each frag or range of frags in
5013  * the csfragino list), allocate a new location and store the frag number
5014  * of that new location in the nfrag field of the csfrag struct.
5015  * If a new frag can't be allocated for any element in the csfragino list,
5016  * set the new frag number for that element to 0 and return immediately.
5017  * The notenoughspace() function will detect this condition.
5018  */
5019 void
5020 alloccsfragino()
5021 {
5022         struct csfrag   *cfp;
5023 
5024         /*
5025          * allocate space for inode frag ranges
5026          */
5027         for (cfp = csfragino; cfp; cfp = cfp->next) {
5028                 allocfrags(cfp->frags, &cfp->nfrag, &cfp->cylno);
5029                 if (cfp->nfrag == 0)
5030                         break;
5031         }
5032 }
5033 
5034 void
5035 alloccsfragfree()
5036 {
5037         struct csfrag   *cfp;
5038 
5039         /*
5040          * allocate the free frags needed for extended summary info
5041          */
5042         rdcg((long)0);
5043 
5044         for (cfp = csfragfree; cfp; cfp = cfp->next)
5045                 clrbit(cg_blksfree(&acg), cfp->ofrag - cgbase(&sblock, 0));
5046 
5047         wtcg();
5048 }
5049 
5050 void
5051 freefrags(daddr32_t frag, long frags, long cylno)
5052 {
5053         int     i;
5054 
5055         /*
5056          * free frags
5057          */
5058         rdcg(cylno);
5059         for (i = 0; i < frags; ++i) {
5060                 setbit(cg_blksfree(&acg), (frag+i) - cgbase(&sblock, cylno));
5061         }
5062         wtcg();
5063 }
5064 
5065 int
5066 findfreerange(long *bitp, long *bitsp)
5067 {
5068         long     bit;
5069 
5070         /*
5071          * find a range of free bits in a cylinder group bit map
5072          */
5073         for (bit = *bitp, *bitsp = 0; bit < acg.cg_ndblk; ++bit)
5074                 if (isset(cg_blksfree(&acg), bit))
5075                         break;
5076 
5077         if (bit >= acg.cg_ndblk)
5078                 return (0);
5079 
5080         *bitp  = bit;
5081         *bitsp = 1;
5082         for (++bit; bit < acg.cg_ndblk; ++bit, ++(*bitsp)) {
5083                 if ((bit % sblock.fs_frag) == 0)
5084                         break;
5085                 if (isclr(cg_blksfree(&acg), bit))
5086                         break;
5087         }
5088         return (1);
5089 }
5090 
5091 void
5092 resetallocinfo()
5093 {
5094         long    cno;
5095         long    bit;
5096         long    bits;
5097 
5098         /*
5099          * Compute the free blocks/frags info and update the appropriate
5100          * inmemory superblock, summary info, and cylinder group fields
5101          */
5102         sblock.fs_cstotal.cs_nffree -= acg.cg_cs.cs_nffree;
5103         sblock.fs_cstotal.cs_nbfree -= acg.cg_cs.cs_nbfree;
5104 
5105         acg.cg_cs.cs_nffree = 0;
5106         acg.cg_cs.cs_nbfree = 0;
5107 
5108         bzero((caddr_t)acg.cg_frsum, sizeof (acg.cg_frsum));
5109         bzero((caddr_t)cg_blktot(&acg), (int)(acg.cg_iusedoff-acg.cg_btotoff));
5110 
5111         bit = bits = 0;
5112         while (findfreerange(&bit, &bits)) {
5113                 if (bits == sblock.fs_frag) {
5114                         acg.cg_cs.cs_nbfree++;
5115                         cno = cbtocylno(&sblock, bit);
5116                         cg_blktot(&acg)[cno]++;
5117                         cg_blks(&sblock, &acg, cno)[cbtorpos(&sblock, bit)]++;
5118                 } else {
5119                         acg.cg_cs.cs_nffree += bits;
5120                         acg.cg_frsum[bits]++;
5121                 }
5122                 bit += bits;
5123         }
5124 
5125         *(fscs + acg.cg_cgx) = acg.cg_cs;
5126 
5127         sblock.fs_cstotal.cs_nffree += acg.cg_cs.cs_nffree;
5128         sblock.fs_cstotal.cs_nbfree += acg.cg_cs.cs_nbfree;
5129 }
5130 
5131 void
5132 extendcg(long cylno)
5133 {
5134         int     i;
5135         diskaddr_t      dupper;
5136         diskaddr_t      cbase;
5137         diskaddr_t      dmax;
5138 
5139         /*
5140          * extend the cylinder group at the end of the old file system
5141          * if it was partially allocated becase of lack of space
5142          */
5143         flcg();
5144         rdcg(cylno);
5145 
5146         dupper = acg.cg_ndblk;
5147         if (cylno == sblock.fs_ncg - 1)
5148                 acg.cg_ncyl = sblock.fs_ncyl - (sblock.fs_cpg * cylno);
5149         else
5150                 acg.cg_ncyl = sblock.fs_cpg;
5151         cbase = cgbase(&sblock, cylno);
5152         dmax = cbase + sblock.fs_fpg;
5153         if (dmax > sblock.fs_size)
5154                 dmax = sblock.fs_size;
5155         acg.cg_ndblk = dmax - cbase;
5156 
5157         for (i = dupper; i < acg.cg_ndblk; ++i)
5158                 setbit(cg_blksfree(&acg), i);
5159 
5160         sblock.fs_dsize += (acg.cg_ndblk - dupper);
5161 
5162         wtcg();
5163         flcg();
5164 }
5165 
5166 struct lockfs   lockfs;
5167 int             lockfd;
5168 int             islocked;
5169 int             lockfskey;
5170 char            lockfscomment[128];
5171 
5172 void
5173 ulockfs()
5174 {
5175         /*
5176          * if the file system was locked, unlock it before exiting
5177          */
5178         if (islocked == 0)
5179                 return;
5180 
5181         /*
5182          * first, check if the lock held
5183          */
5184         lockfs.lf_flags = LOCKFS_MOD;
5185         if (ioctl(lockfd, _FIOLFSS, &lockfs) == -1) {
5186                 perror(directory);
5187                 lockexit(32);
5188         }
5189 
5190         if (LOCKFS_IS_MOD(&lockfs)) {
5191                 (void) fprintf(stderr,
5192                     gettext("FILE SYSTEM CHANGED DURING GROWFS!\n"));
5193                 (void) fprintf(stderr,
5194                     gettext("   See lockfs(1), umount(1), and fsck(1)\n"));
5195                 lockexit(32);
5196         }
5197         /*
5198          * unlock the file system
5199          */
5200         lockfs.lf_lock  = LOCKFS_ULOCK;
5201         lockfs.lf_flags = 0;
5202         lockfs.lf_key   = lockfskey;
5203         clockfs();
5204         if (ioctl(lockfd, _FIOLFS, &lockfs) == -1) {
5205                 perror(directory);
5206                 lockexit(32);
5207         }
5208 }
5209 
5210 void
5211 wlockfs()
5212 {
5213 
5214         /*
5215          * if no-write (-N), don't bother
5216          */
5217         if (Nflag)
5218                 return;
5219         /*
5220          * open the mountpoint, and write lock the file system
5221          */
5222         if ((lockfd = open64(directory, O_RDONLY)) == -1) {
5223                 perror(directory);
5224                 lockexit(32);
5225         }
5226 
5227         /*
5228          * check if it is already locked
5229          */
5230         if (ioctl(lockfd, _FIOLFSS, &lockfs) == -1) {
5231                 perror(directory);
5232                 lockexit(32);
5233         }
5234 
5235         if (lockfs.lf_lock != LOCKFS_WLOCK) {
5236                 lockfs.lf_lock  = LOCKFS_WLOCK;
5237                 lockfs.lf_flags = 0;
5238                 lockfs.lf_key   = 0;
5239                 clockfs();
5240                 if (ioctl(lockfd, _FIOLFS, &lockfs) == -1) {
5241                         perror(directory);
5242                         lockexit(32);
5243                 }
5244         }
5245         islocked = 1;
5246         lockfskey = lockfs.lf_key;
5247 }
5248 
5249 void
5250 clockfs()
5251 {
5252         time_t  t;
5253         char    *ct;
5254 
5255         (void) time(&t);
5256         ct = ctime(&t);
5257         ct[strlen(ct)-1] = '\0';
5258 
5259         (void) sprintf(lockfscomment, "%s -- mkfs pid %d", ct, getpid());
5260         lockfs.lf_comlen  = strlen(lockfscomment)+1;
5261         lockfs.lf_comment = lockfscomment;
5262 }
5263 
5264 /*
5265  * Write the csum records and the superblock
5266  */
5267 void
5268 wtsb()
5269 {
5270         long    i;
5271 
5272         /*
5273          * write summary information
5274          */
5275         for (i = 0; i < sblock.fs_cssize; i += sblock.fs_bsize)
5276                 wtfs(fsbtodb(&sblock, (uint64_t)(sblock.fs_csaddr +
5277                     numfrags(&sblock, i))),
5278                     (int)(sblock.fs_cssize - i < sblock.fs_bsize ?
5279                     sblock.fs_cssize - i : sblock.fs_bsize),
5280                     ((char *)fscs) + i);
5281 
5282         /*
5283          * write superblock
5284          */
5285         sblock.fs_time = mkfstime;
5286         wtfs((diskaddr_t)(SBOFF / sectorsize), sbsize, (char *)&sblock);
5287 }
5288 
5289 /*
5290  * Verify that the optimization selection is reasonable, and advance
5291  * the global "string" appropriately.
5292  */
5293 static char
5294 checkopt(char *optim)
5295 {
5296         char    opt;
5297         int     limit = strcspn(optim, ",");
5298 
5299         switch (limit) {
5300         case 0: /* missing indicator (have comma or nul) */
5301                 (void) fprintf(stderr, gettext(
5302                     "mkfs: missing optimization flag reset to `t' (time)\n"));
5303                 opt = 't';
5304                 break;
5305 
5306         case 1: /* single-character indicator */
5307                 opt = *optim;
5308                 if ((opt != 's') && (opt != 't')) {
5309                         (void) fprintf(stderr, gettext(
5310                     "mkfs: bad optimization value `%c' reset to `t' (time)\n"),
5311                             opt);
5312                         opt = 't';
5313                 }
5314                 break;
5315 
5316         default: /* multi-character indicator */
5317                 (void) fprintf(stderr, gettext(
5318             "mkfs: bad optimization value `%*.*s' reset to `t' (time)\n"),
5319                     limit, limit, optim);
5320                 opt = 't';
5321                 break;
5322         }
5323 
5324         string += limit;
5325 
5326         return (opt);
5327 }
5328 
5329 /*
5330  * Verify that the mtb selection is reasonable, and advance
5331  * the global "string" appropriately.
5332  */
5333 static char
5334 checkmtb(char *mtbarg)
5335 {
5336         char    mtbc;
5337         int     limit = strcspn(mtbarg, ",");
5338 
5339         switch (limit) {
5340         case 0: /* missing indicator (have comma or nul) */
5341                 (void) fprintf(stderr, gettext(
5342                     "mkfs: missing mtb flag reset to `n' (no mtb support)\n"));
5343                 mtbc = 'n';
5344                 break;
5345 
5346         case 1: /* single-character indicator */
5347                 mtbc = tolower(*mtbarg);
5348                 if ((mtbc != 'y') && (mtbc != 'n')) {
5349                         (void) fprintf(stderr, gettext(
5350                     "mkfs: bad mtb value `%c' reset to `n' (no mtb support)\n"),
5351                             mtbc);
5352                         mtbc = 'n';
5353                 }
5354                 break;
5355 
5356         default: /* multi-character indicator */
5357                 (void) fprintf(stderr, gettext(
5358             "mkfs: bad mtb value `%*.*s' reset to `n' (no mtb support)\n"),
5359                     limit, limit, mtbarg);
5360                 opt = 'n';
5361                 break;
5362         }
5363 
5364         string += limit;
5365 
5366         return (mtbc);
5367 }
5368 
5369 /*
5370  * Verify that a value is in a range.  If it is not, resets it to
5371  * its default value if one is supplied, exits otherwise.
5372  *
5373  * When testing, can compare user_supplied to RC_KEYWORD or RC_POSITIONAL.
5374  */
5375 static void
5376 range_check(long *varp, char *name, long minimum, long maximum,
5377     long def_val, int user_supplied)
5378 {
5379         dprintf(("DeBuG %s : %ld (%ld %ld %ld)\n",
5380             name, *varp, minimum, maximum, def_val));
5381 
5382         if ((*varp < minimum) || (*varp > maximum)) {
5383                 if (user_supplied != RC_DEFAULT) {
5384                         (void) fprintf(stderr, gettext(
5385             "mkfs: bad value for %s: %ld must be between %ld and %ld\n"),
5386                             name, *varp, minimum, maximum);
5387                 }
5388                 if (def_val != NO_DEFAULT) {
5389                         if (user_supplied) {
5390                                 (void) fprintf(stderr,
5391                                     gettext("mkfs: %s reset to default %ld\n"),
5392                                     name, def_val);
5393                         }
5394                         *varp = def_val;
5395                         dprintf(("DeBuG %s : %ld\n", name, *varp));
5396                         return;
5397                 }
5398                 lockexit(2);
5399                 /*NOTREACHED*/
5400         }
5401 }
5402 
5403 /*
5404  * Verify that a value is in a range.  If it is not, resets it to
5405  * its default value if one is supplied, exits otherwise.
5406  *
5407  * When testing, can compare user_supplied to RC_KEYWORD or RC_POSITIONAL.
5408  */
5409 static void
5410 range_check_64(uint64_t *varp, char *name, uint64_t minimum, uint64_t maximum,
5411     uint64_t def_val, int user_supplied)
5412 {
5413         if ((*varp < minimum) || (*varp > maximum)) {
5414                 if (user_supplied != RC_DEFAULT) {
5415                         (void) fprintf(stderr, gettext(
5416             "mkfs: bad value for %s: %lld must be between %lld and %lld\n"),
5417                             name, *varp, minimum, maximum);
5418                 }
5419                 if (def_val != NO_DEFAULT) {
5420                         if (user_supplied) {
5421                                 (void) fprintf(stderr,
5422                                     gettext("mkfs: %s reset to default %lld\n"),
5423                                     name, def_val);
5424                         }
5425                         *varp = def_val;
5426                         return;
5427                 }
5428                 lockexit(2);
5429                 /*NOTREACHED*/
5430         }
5431 }
5432 
5433 /*
5434  * Blocks SIGINT from delivery.  Returns the previous mask in the
5435  * buffer provided, so that mask may be later restored.
5436  */
5437 static void
5438 block_sigint(sigset_t *old_mask)
5439 {
5440         sigset_t block_mask;
5441 
5442         if (sigemptyset(&block_mask) < 0) {
5443                 fprintf(stderr, gettext("Could not clear signal mask\n"));
5444                 lockexit(3);
5445         }
5446         if (sigaddset(&block_mask, SIGINT) < 0) {
5447                 fprintf(stderr, gettext("Could not set signal mask\n"));
5448                 lockexit(3);
5449         }
5450         if (sigprocmask(SIG_BLOCK, &block_mask, old_mask) < 0) {
5451                 fprintf(stderr, gettext("Could not block SIGINT\n"));
5452                 lockexit(3);
5453         }
5454 }
5455 
5456 /*
5457  * Restores the signal mask that was in force before a call
5458  * to block_sigint().  This may actually still have SIGINT blocked,
5459  * if we've been recursively invoked.
5460  */
5461 static void
5462 unblock_sigint(sigset_t *old_mask)
5463 {
5464         if (sigprocmask(SIG_UNBLOCK, old_mask, (sigset_t *)NULL) < 0) {
5465                 fprintf(stderr, gettext("Could not restore signal mask\n"));
5466                 lockexit(3);
5467         }
5468 }
5469 
5470 /*
5471  * Attempt to be somewhat graceful about being interrupted, rather than
5472  * just silently leaving the filesystem in an unusable state.
5473  *
5474  * The kernel has blocked SIGINT upon entry, so we don't have to worry
5475  * about recursion if the user starts pounding on the keyboard.
5476  */
5477 static void
5478 recover_from_sigint(int signum)
5479 {
5480         if (fso > -1) {
5481                 if ((Nflag != 0) || confirm_abort()) {
5482                         lockexit(4);
5483                 }
5484         }
5485 }
5486 
5487 static int
5488 confirm_abort(void)
5489 {
5490         char line[80];
5491 
5492         printf(gettext("\n\nAborting at this point will leave the filesystem "
5493             "in an inconsistent\nstate.  If you do choose to stop, "
5494             "you will be given instructions on how to\nrecover "
5495             "the filesystem.  Do you wish to cancel the filesystem "
5496             "grow\noperation (y/n)?"));
5497         if (getaline(stdin, line, sizeof (line)) == EOF)
5498                 line[0] = 'y';
5499 
5500         printf("\n");
5501         if (line[0] == 'y' || line[0] == 'Y')
5502                 return (1);
5503         else {
5504                 return (0);
5505         }
5506 }
5507 
5508 static int
5509 getaline(FILE *fp, char *loc, int maxlen)
5510 {
5511         int n;
5512         char *p, *lastloc;
5513 
5514         p = loc;
5515         lastloc = &p[maxlen-1];
5516         while ((n = getc(fp)) != '\n') {
5517                 if (n == EOF)
5518                         return (EOF);
5519                 if (!isspace(n) && p < lastloc)
5520                         *p++ = n;
5521         }
5522         *p = 0;
5523         return (p - loc);
5524 }
5525 
5526 /*
5527  * Calculate the maximum value of cylinders-per-group for a file
5528  * system with the characteristics:
5529  *
5530  *      bsize - file system block size
5531  *      fragsize - frag size
5532  *      nbpi - number of bytes of disk space per inode
5533  *      nrpos - number of rotational positions
5534  *      spc - sectors per cylinder
5535  *
5536  * These five characteristic are not adjustable (by this function).
5537  * The only attribute of the file system which IS adjusted by this
5538  * function in order to maximize cylinders-per-group is the proportion
5539  * of the cylinder group overhead block used for the inode map.  The
5540  * inode map cannot occupy more than one-third of the cylinder group
5541  * overhead block, but it's OK for it to occupy less than one-third
5542  * of the overhead block.
5543  *
5544  * The setting of nbpi determines one possible value for the maximum
5545  * size of a cylinder group.  It does so because it determines the total
5546  * number of inodes in the file system (file system size is fixed, and
5547  * nbpi is fixed, so the total number of inodes is fixed too).  The
5548  * cylinder group has to be small enough so that the number of inodes
5549  * in the cylinder group is less than or equal to the number of bits
5550  * in one-third (or whatever proportion is assumed) of a file system
5551  * block.  The details of the calculation are:
5552  *
5553  *     The macro MAXIpG_B(bsize, inode_divisor) determines the maximum
5554  *     number of inodes that can be in a cylinder group, given the
5555  *     proportion of the cylinder group overhead block used for the
5556  *     inode bitmaps (an inode_divisor of 3 means that 1/3 of the
5557  *     block is used for inode bitmaps; an inode_divisor of 12 means
5558  *     that 1/12 of the block is used for inode bitmaps.)
5559  *
5560  *     Once the number of inodes per cylinder group is known, the
5561  *     maximum value of cylinders-per-group (determined by nbpi)
5562  *     is calculated by the formula
5563  *
5564  *     maxcpg_given_nbpi = (size of a cylinder group)/(size of a cylinder)
5565  *
5566  *                       = (inodes-per-cg * nbpi)/(spc * DEV_BSIZE)
5567  *
5568  *     (Interestingly, the size of the file system never enters
5569  *     into this calculation.)
5570  *
5571  * Another possible value for the maximum cylinder group size is determined
5572  * by frag_size and nrpos.  The frags in the cylinder group must be
5573  * representable in the frag bitmaps in the cylinder overhead block and the
5574  * rotational positions for each cylinder must be represented in the
5575  * rotational position tables.  The calculation of the maximum cpg
5576  * value, given the frag and nrpos vales, is:
5577  *
5578  *     maxcpg_given_fragsize =
5579  *        (available space in the overhead block) / (size of per-cylinder data)
5580  *
5581  *     The available space in the overhead block =
5582  *        bsize - sizeof (struct cg) - space_used_for_inode_bitmaps
5583  *
5584  *     The size of the per-cylinder data is:
5585  *          sizeof(long)            # for the "blocks avail per cylinder" field
5586  *          + nrpos * sizeof(short)   # for the rotational position table entry
5587  *          + frags-per-cylinder/NBBY # number of bytes to represent this
5588  *                                    # cylinder in the frag bitmap
5589  *
5590  * The two calculated maximum values of cylinder-per-group will typically
5591  * turn out to be different, since they are derived from two different
5592  * constraints.  Usually, maxcpg_given_nbpi is much bigger than
5593  * maxcpg_given_fragsize.  But they can be brought together by
5594  * adjusting the proportion of the overhead block dedicated to
5595  * the inode bitmaps.  Decreasing the proportion of the cylinder
5596  * group overhead block used for inode maps will decrease
5597  * maxcpg_given_nbpi and increase maxcpg_given_fragsize.
5598  *
5599  * This function calculates the initial values of maxcpg_given_nbpi
5600  * and maxcpg_given_fragsize assuming that 1/3 of the cg overhead
5601  * block is used for inode bitmaps.  Then it decreases the proportion
5602  * of the cg overhead block used for inode bitmaps (by increasing
5603  * the value of inode_divisor) until maxcpg_given_nbpi and
5604  * maxcpg_given_fragsize are the same, or stop changing, or
5605  * maxcpg_given_nbpi is less than maxcpg_given_fragsize.
5606  *
5607  * The loop terminates when any of the following occur:
5608  *      * maxcpg_given_fragsize is greater than or equal to
5609  *        maxcpg_given_nbpi
5610  *      * neither maxcpg_given_fragsize nor maxcpg_given_nbpi
5611  *        change in the expected direction
5612  *
5613  * The loop is guaranteed to terminate because it only continues
5614  * while maxcpg_given_fragsize and maxcpg_given_nbpi are approaching
5615  * each other.  As soon they cross each other, or neither one changes
5616  * in the direction of the other, or one of them moves in the wrong
5617  * direction, the loop completes.
5618  */
5619 
5620 static long
5621 compute_maxcpg(long bsize, long fragsize, long nbpi, long nrpos, long spc)
5622 {
5623         int     maxcpg_given_nbpi;      /* in cylinders */
5624         int     maxcpg_given_fragsize;  /* in cylinders */
5625         int     spf;                    /* sectors per frag */
5626         int     inode_divisor;
5627         int     old_max_given_frag = 0;
5628         int     old_max_given_nbpi = INT_MAX;
5629 
5630         spf = fragsize / DEV_BSIZE;
5631         inode_divisor = 3;
5632 
5633         while (1) {
5634                 maxcpg_given_nbpi =
5635                     (((int64_t)(MAXIpG_B(bsize, inode_divisor))) * nbpi) /
5636                     (DEV_BSIZE * ((int64_t)spc));
5637                 maxcpg_given_fragsize =
5638                     (bsize - (sizeof (struct cg)) - (bsize / inode_divisor)) /
5639                     (sizeof (long) + nrpos * sizeof (short) +
5640                     (spc / spf) / NBBY);
5641 
5642                 if (maxcpg_given_fragsize >= maxcpg_given_nbpi)
5643                         return (maxcpg_given_nbpi);
5644 
5645                 /*
5646                  * If neither value moves toward the other, return the
5647                  * least of the old values (we use the old instead of the
5648                  * new because: if the old is the same as the new, it
5649                  * doesn't matter which ones we use.  If one of the
5650                  * values changed, but in the wrong direction, the
5651                  * new values are suspect.  Better use the old.  This
5652                  * shouldn't happen, but it's best to check.
5653                  */
5654 
5655                 if (!(maxcpg_given_nbpi < old_max_given_nbpi) &&
5656                     !(maxcpg_given_fragsize > old_max_given_frag))
5657                         return (MIN(old_max_given_nbpi, old_max_given_frag));
5658 
5659                 /*
5660                  * This is probably impossible, but if one of the maxcpg
5661                  * values moved in the "right" direction and one moved
5662                  * in the "wrong" direction (that is, the two values moved
5663                  * in the same direction), the previous conditional won't
5664                  * recognize that the values aren't converging (since at
5665                  * least one value moved in the "right" direction, the
5666                  * last conditional says "keep going").
5667                  *
5668                  * Just to make absolutely certain that the loop terminates,
5669                  * check for one of the values moving in the "wrong" direction
5670                  * and terminate the loop if it happens.
5671                  */
5672 
5673                 if (maxcpg_given_nbpi > old_max_given_nbpi ||
5674                     maxcpg_given_fragsize < old_max_given_frag)
5675                         return (MIN(old_max_given_nbpi, old_max_given_frag));
5676 
5677                 old_max_given_nbpi = maxcpg_given_nbpi;
5678                 old_max_given_frag = maxcpg_given_fragsize;
5679 
5680                 inode_divisor++;
5681         }
5682 }
5683 
5684 static int
5685 in_64bit_mode(void)
5686 {
5687         /*  cmd must be an absolute path, for security */
5688         char *cmd = "/usr/bin/isainfo -b";
5689         char buf[BUFSIZ];
5690         FILE *ptr;
5691         int retval = 0;
5692 
5693         putenv("IFS= \t");
5694         if ((ptr = popen(cmd, "r")) != NULL) {
5695                 if (fgets(buf, BUFSIZ, ptr) != NULL &&
5696                     strncmp(buf, "64", 2) == 0)
5697                         retval = 1;
5698                 (void) pclose(ptr);
5699         }
5700         return (retval);
5701 }
5702 
5703 /*
5704  * validate_size
5705  *
5706  * Return 1 if the device appears to be at least "size" sectors long.
5707  * Return 0 if it's shorter or we can't read it.
5708  */
5709 
5710 static int
5711 validate_size(int fd, diskaddr_t size)
5712 {
5713         char    buf[DEV_BSIZE];
5714         int rc;
5715 
5716         if ((llseek(fd, (offset_t)((size - 1) * DEV_BSIZE), SEEK_SET) == -1) ||
5717             (read(fd, buf, DEV_BSIZE)) != DEV_BSIZE)
5718                 rc = 0;
5719         else
5720                 rc = 1;
5721         return (rc);
5722 }
5723 
5724 /*
5725  * Print every field of the calculated superblock, along with
5726  * its value.  To make parsing easier on the caller, the value
5727  * is printed first, then the name.  Additionally, there's only
5728  * one name/value pair per line.  All values are reported in
5729  * hexadecimal (with the traditional 0x prefix), as that's slightly
5730  * easier for humans to read.  Not that they're expected to, but
5731  * debugging happens.
5732  */
5733 static void
5734 dump_sblock(void)
5735 {
5736         int row, column, pending, written;
5737         caddr_t source;
5738 
5739         if (Rflag) {
5740                 pending = sizeof (sblock);
5741                 source = (caddr_t)&sblock;
5742                 do {
5743                         written = write(fileno(stdout), source, pending);
5744                         pending -= written;
5745                         source += written;
5746                 } while ((pending > 0) && (written > 0));
5747 
5748                 if (written < 0) {
5749                         perror(gettext("Binary dump of superblock failed"));
5750                         lockexit(1);
5751                 }
5752                 return;
5753         } else {
5754                 printf("0x%x sblock.fs_link\n", sblock.fs_link);
5755                 printf("0x%x sblock.fs_rolled\n", sblock.fs_rolled);
5756                 printf("0x%x sblock.fs_sblkno\n", sblock.fs_sblkno);
5757                 printf("0x%x sblock.fs_cblkno\n", sblock.fs_cblkno);
5758                 printf("0x%x sblock.fs_iblkno\n", sblock.fs_iblkno);
5759                 printf("0x%x sblock.fs_dblkno\n", sblock.fs_dblkno);
5760                 printf("0x%x sblock.fs_cgoffset\n", sblock.fs_cgoffset);
5761                 printf("0x%x sblock.fs_cgmask\n", sblock.fs_cgmask);
5762                 printf("0x%x sblock.fs_time\n", sblock.fs_time);
5763                 printf("0x%x sblock.fs_size\n", sblock.fs_size);
5764                 printf("0x%x sblock.fs_dsize\n", sblock.fs_dsize);
5765                 printf("0x%x sblock.fs_ncg\n", sblock.fs_ncg);
5766                 printf("0x%x sblock.fs_bsize\n", sblock.fs_bsize);
5767                 printf("0x%x sblock.fs_fsize\n", sblock.fs_fsize);
5768                 printf("0x%x sblock.fs_frag\n", sblock.fs_frag);
5769                 printf("0x%x sblock.fs_minfree\n", sblock.fs_minfree);
5770                 printf("0x%x sblock.fs_rotdelay\n", sblock.fs_rotdelay);
5771                 printf("0x%x sblock.fs_rps\n", sblock.fs_rps);
5772                 printf("0x%x sblock.fs_bmask\n", sblock.fs_bmask);
5773                 printf("0x%x sblock.fs_fmask\n", sblock.fs_fmask);
5774                 printf("0x%x sblock.fs_bshift\n", sblock.fs_bshift);
5775                 printf("0x%x sblock.fs_fshift\n", sblock.fs_fshift);
5776                 printf("0x%x sblock.fs_maxcontig\n", sblock.fs_maxcontig);
5777                 printf("0x%x sblock.fs_maxbpg\n", sblock.fs_maxbpg);
5778                 printf("0x%x sblock.fs_fragshift\n", sblock.fs_fragshift);
5779                 printf("0x%x sblock.fs_fsbtodb\n", sblock.fs_fsbtodb);
5780                 printf("0x%x sblock.fs_sbsize\n", sblock.fs_sbsize);
5781                 printf("0x%x sblock.fs_csmask\n", sblock.fs_csmask);
5782                 printf("0x%x sblock.fs_csshift\n", sblock.fs_csshift);
5783                 printf("0x%x sblock.fs_nindir\n", sblock.fs_nindir);
5784                 printf("0x%x sblock.fs_inopb\n", sblock.fs_inopb);
5785                 printf("0x%x sblock.fs_nspf\n", sblock.fs_nspf);
5786                 printf("0x%x sblock.fs_optim\n", sblock.fs_optim);
5787 #ifdef _LITTLE_ENDIAN
5788                 printf("0x%x sblock.fs_state\n", sblock.fs_state);
5789 #else
5790                 printf("0x%x sblock.fs_npsect\n", sblock.fs_npsect);
5791 #endif
5792                 printf("0x%x sblock.fs_si\n", sblock.fs_si);
5793                 printf("0x%x sblock.fs_trackskew\n", sblock.fs_trackskew);
5794                 printf("0x%x sblock.fs_id[0]\n", sblock.fs_id[0]);
5795                 printf("0x%x sblock.fs_id[1]\n", sblock.fs_id[1]);
5796                 printf("0x%x sblock.fs_csaddr\n", sblock.fs_csaddr);
5797                 printf("0x%x sblock.fs_cssize\n", sblock.fs_cssize);
5798                 printf("0x%x sblock.fs_cgsize\n", sblock.fs_cgsize);
5799                 printf("0x%x sblock.fs_ntrak\n", sblock.fs_ntrak);
5800                 printf("0x%x sblock.fs_nsect\n", sblock.fs_nsect);
5801                 printf("0x%x sblock.fs_spc\n", sblock.fs_spc);
5802                 printf("0x%x sblock.fs_ncyl\n", sblock.fs_ncyl);
5803                 printf("0x%x sblock.fs_cpg\n", sblock.fs_cpg);
5804                 printf("0x%x sblock.fs_ipg\n", sblock.fs_ipg);
5805                 printf("0x%x sblock.fs_fpg\n", sblock.fs_fpg);
5806                 printf("0x%x sblock.fs_cstotal\n", sblock.fs_cstotal);
5807                 printf("0x%x sblock.fs_fmod\n", sblock.fs_fmod);
5808                 printf("0x%x sblock.fs_clean\n", sblock.fs_clean);
5809                 printf("0x%x sblock.fs_ronly\n", sblock.fs_ronly);
5810                 printf("0x%x sblock.fs_flags\n", sblock.fs_flags);
5811                 printf("0x%x sblock.fs_fsmnt\n", sblock.fs_fsmnt);
5812                 printf("0x%x sblock.fs_cgrotor\n", sblock.fs_cgrotor);
5813                 printf("0x%x sblock.fs_u.fs_csp\n", sblock.fs_u.fs_csp);
5814                 printf("0x%x sblock.fs_cpc\n", sblock.fs_cpc);
5815 
5816                 /*
5817                  * No macros are defined for the dimensions of the
5818                  * opostbl array.
5819                  */
5820                 for (row = 0; row < 16; row++) {
5821                         for (column = 0; column < 8; column++) {
5822                                 printf("0x%x sblock.fs_opostbl[%d][%d]\n",
5823                                     sblock.fs_opostbl[row][column],
5824                                     row, column);
5825                         }
5826                 }
5827 
5828                 /*
5829                  * Ditto the size of sparecon.
5830                  */
5831                 for (row = 0; row < 51; row++) {
5832                         printf("0x%x sblock.fs_sparecon[%d]\n",
5833                             sblock.fs_sparecon[row], row);
5834                 }
5835 
5836                 printf("0x%x sblock.fs_version\n", sblock.fs_version);
5837                 printf("0x%x sblock.fs_logbno\n", sblock.fs_logbno);
5838                 printf("0x%x sblock.fs_reclaim\n", sblock.fs_reclaim);
5839                 printf("0x%x sblock.fs_sparecon2\n", sblock.fs_sparecon2);
5840 #ifdef _LITTLE_ENDIAN
5841                 printf("0x%x sblock.fs_npsect\n", sblock.fs_npsect);
5842 #else
5843                 printf("0x%x sblock.fs_state\n", sblock.fs_state);
5844 #endif
5845                 printf("0x%llx sblock.fs_qbmask\n", sblock.fs_qbmask);
5846                 printf("0x%llx sblock.fs_qfmask\n", sblock.fs_qfmask);
5847                 printf("0x%x sblock.fs_postblformat\n", sblock.fs_postblformat);
5848                 printf("0x%x sblock.fs_nrpos\n", sblock.fs_nrpos);
5849                 printf("0x%x sblock.fs_postbloff\n", sblock.fs_postbloff);
5850                 printf("0x%x sblock.fs_rotbloff\n", sblock.fs_rotbloff);
5851                 printf("0x%x sblock.fs_magic\n", sblock.fs_magic);
5852 
5853                 /*
5854                  * fs_space isn't of much use in this context, so we'll
5855                  * just ignore it for now.
5856                  */
5857         }
5858 }
--- EOF ---