121184Sdist /* 221184Sdist * Copyright (c) 1983 Regents of the University of California. 321184Sdist * All rights reserved. The Berkeley software License Agreement 421184Sdist * specifies the terms and conditions for redistribution. 521184Sdist */ 621184Sdist 711288Smckusick #ifndef lint 821184Sdist char copyright[] = 921184Sdist "@(#) Copyright (c) 1983 Regents of the University of California.\n\ 1021184Sdist All rights reserved.\n"; 1121184Sdist #endif not lint 1211288Smckusick 1321184Sdist #ifndef lint 14*24700Smckusick static char sccsid[] = "@(#)tunefs.c 5.2 (Berkeley) 09/11/85"; 1521184Sdist #endif not lint 1621184Sdist 1711288Smckusick /* 1811288Smckusick * tunefs: change layout parameters to an existing file system. 1911288Smckusick */ 2011288Smckusick 2111288Smckusick #include <sys/param.h> 2211288Smckusick #include <sys/stat.h> 2311288Smckusick #include <sys/fs.h> 2411288Smckusick #include <sys/inode.h> 2511288Smckusick 2613120Ssam #include <stdio.h> 2713120Ssam #include <fstab.h> 2813120Ssam 2911288Smckusick union { 3011288Smckusick struct fs sb; 3111288Smckusick char pad[MAXBSIZE]; 3211288Smckusick } sbun; 3311288Smckusick #define sblock sbun.sb 3411288Smckusick 3511288Smckusick int fi; 3611288Smckusick 3711288Smckusick main(argc, argv) 3811288Smckusick int argc; 3911288Smckusick char *argv[]; 4011288Smckusick { 4111288Smckusick char *cp, *special, *name; 4211288Smckusick struct stat st; 4311288Smckusick int i; 4411288Smckusick int Aflag = 0; 45*24700Smckusick struct fstab *fs; 46*24700Smckusick char *chg[2], device[MAXPATHLEN]; 4711288Smckusick extern char *sprintf(); 4811288Smckusick 4911288Smckusick argc--, argv++; 5011288Smckusick if (argc < 2) 5111288Smckusick goto usage; 5211288Smckusick special = argv[argc - 1]; 5313120Ssam fs = getfsfile(special); 5413120Ssam if (fs) 5513120Ssam special = fs->fs_spec; 5611288Smckusick again: 5711288Smckusick if (stat(special, &st) < 0) { 5811288Smckusick if (*special != '/') { 5911288Smckusick if (*special == 'r') 6011288Smckusick special++; 6111288Smckusick special = sprintf(device, "/dev/%s", special); 6211288Smckusick goto again; 6311288Smckusick } 6411288Smckusick fprintf(stderr, "tunefs: "); perror(special); 6511288Smckusick exit(1); 6611288Smckusick } 6711288Smckusick if ((st.st_mode & S_IFMT) != S_IFBLK && 6811288Smckusick (st.st_mode & S_IFMT) != S_IFCHR) 6911288Smckusick fatal("%s: not a block or character device", special); 7011288Smckusick getsb(&sblock, special); 7111288Smckusick for (; argc > 0 && argv[0][0] == '-'; argc--, argv++) { 7211288Smckusick for (cp = &argv[0][1]; *cp; cp++) 7311288Smckusick switch (*cp) { 7411288Smckusick 7511288Smckusick case 'A': 7611288Smckusick Aflag++; 7711288Smckusick continue; 7811288Smckusick 7911288Smckusick case 'a': 8011288Smckusick name = "maximum contiguous block count"; 8111288Smckusick if (argc < 1) 8211288Smckusick fatal("-a: missing %s", name); 8311288Smckusick argc--, argv++; 8411288Smckusick i = atoi(*argv); 8511288Smckusick if (i < 1) 8611288Smckusick fatal("%s: %s must be >= 1", 8711288Smckusick *argv, name); 8811288Smckusick fprintf(stdout, "%s changes from %d to %d\n", 8911288Smckusick name, sblock.fs_maxcontig, i); 9011288Smckusick sblock.fs_maxcontig = i; 9111288Smckusick continue; 9211288Smckusick 9311288Smckusick case 'd': 9411288Smckusick name = 9511288Smckusick "rotational delay between contiguous blocks"; 9611288Smckusick if (argc < 1) 9711288Smckusick fatal("-d: missing %s", name); 9811288Smckusick argc--, argv++; 9911288Smckusick i = atoi(*argv); 10011288Smckusick if (i < 0) 10111288Smckusick fatal("%s: bad %s", *argv, name); 10211288Smckusick fprintf(stdout, 10311288Smckusick "%s changes from %dms to %dms\n", 10411288Smckusick name, sblock.fs_rotdelay, i); 10511288Smckusick sblock.fs_rotdelay = i; 10611288Smckusick continue; 10711288Smckusick 10811288Smckusick case 'e': 10911288Smckusick name = 11011288Smckusick "maximum blocks per file in a cylinder group"; 11111288Smckusick if (argc < 1) 11211288Smckusick fatal("-e: missing %s", name); 11311288Smckusick argc--, argv++; 11411288Smckusick i = atoi(*argv); 11511288Smckusick if (i < 1) 11611288Smckusick fatal("%s: %s must be >= 1", 11711288Smckusick *argv, name); 11811288Smckusick fprintf(stdout, "%s changes from %d to %d\n", 11911288Smckusick name, sblock.fs_maxbpg, i); 12011288Smckusick sblock.fs_maxbpg = i; 12111288Smckusick continue; 12211288Smckusick 12311288Smckusick case 'm': 12411288Smckusick name = "minimum percentage of free space"; 12511288Smckusick if (argc < 1) 12611288Smckusick fatal("-m: missing %s", name); 12711288Smckusick argc--, argv++; 12811288Smckusick i = atoi(*argv); 12911288Smckusick if (i < 0 || i > 99) 13011288Smckusick fatal("%s: bad %s", *argv, name); 13111288Smckusick fprintf(stdout, 13211288Smckusick "%s changes from %d%% to %d%%\n", 13311288Smckusick name, sblock.fs_minfree, i); 13411288Smckusick sblock.fs_minfree = i; 13511288Smckusick continue; 13611288Smckusick 137*24700Smckusick case 'o': 138*24700Smckusick name = "optimization preference"; 139*24700Smckusick if (argc < 1) 140*24700Smckusick fatal("-o: missing %s", name); 141*24700Smckusick argc--, argv++; 142*24700Smckusick chg[FS_OPTSPACE] = "space"; 143*24700Smckusick chg[FS_OPTTIME] = "time"; 144*24700Smckusick if (strcmp(*argv, chg[FS_OPTSPACE]) == 0) 145*24700Smckusick i = FS_OPTSPACE; 146*24700Smckusick else if (strcmp(*argv, chg[FS_OPTTIME]) == 0) 147*24700Smckusick i = FS_OPTTIME; 148*24700Smckusick else 149*24700Smckusick fatal("%s: bad %s (options are `space' or `time')", 150*24700Smckusick *argv, name); 151*24700Smckusick if (sblock.fs_optim == i) { 152*24700Smckusick fprintf(stdout, 153*24700Smckusick "%s remains unchanged as %s\n", 154*24700Smckusick name, chg[i]); 155*24700Smckusick continue; 156*24700Smckusick } 157*24700Smckusick fprintf(stdout, 158*24700Smckusick "%s changes from %s to %s\n", 159*24700Smckusick name, chg[sblock.fs_optim], chg[i]); 160*24700Smckusick sblock.fs_optim = i; 161*24700Smckusick continue; 162*24700Smckusick 16311288Smckusick default: 16411288Smckusick fatal("-%c: unknown flag", *cp); 16511288Smckusick } 16611288Smckusick } 16711288Smckusick if (argc != 1) 16811288Smckusick goto usage; 16911288Smckusick bwrite(SBLOCK, (char *)&sblock, SBSIZE); 17011288Smckusick if (Aflag) 17111288Smckusick for (i = 0; i < sblock.fs_ncg; i++) 17211288Smckusick bwrite(fsbtodb(&sblock, cgsblock(&sblock, i)), 17311288Smckusick (char *)&sblock, SBSIZE); 17411288Smckusick close(fi); 17511288Smckusick exit(0); 17611288Smckusick usage: 17711288Smckusick fprintf(stderr, "Usage: tunefs tuneup-options special-device\n"); 17811288Smckusick fprintf(stderr, "where tuneup-options are:\n"); 17913555Ssam fprintf(stderr, "\t-a maximum contiguous blocks\n"); 18011288Smckusick fprintf(stderr, "\t-d rotational delay between contiguous blocks\n"); 18111288Smckusick fprintf(stderr, "\t-e maximum blocks per file in a cylinder group\n"); 18211288Smckusick fprintf(stderr, "\t-m minimum percentage of free space\n"); 183*24700Smckusick fprintf(stderr, "\t-o optimization preference (`space' or `time')\n"); 18411288Smckusick exit(2); 18511288Smckusick } 18611288Smckusick 18711288Smckusick getsb(fs, file) 18811288Smckusick register struct fs *fs; 18911288Smckusick char *file; 19011288Smckusick { 19111288Smckusick 19211288Smckusick fi = open(file, 2); 19311288Smckusick if (fi < 0) { 19411288Smckusick fprintf(stderr, "cannot open"); 19511288Smckusick perror(file); 19611288Smckusick exit(3); 19711288Smckusick } 19811288Smckusick if (bread(SBLOCK, (char *)fs, SBSIZE)) { 19911288Smckusick fprintf(stderr, "bad super block"); 20011288Smckusick perror(file); 20111288Smckusick exit(4); 20211288Smckusick } 20311288Smckusick if (fs->fs_magic != FS_MAGIC) { 20411288Smckusick fprintf(stderr, "%s: bad magic number\n", file); 20511288Smckusick exit(5); 20611288Smckusick } 20711288Smckusick } 20811288Smckusick 20911288Smckusick bwrite(blk, buf, size) 21011288Smckusick char *buf; 21111288Smckusick daddr_t blk; 21211288Smckusick register size; 21311288Smckusick { 21411288Smckusick if (lseek(fi, blk * DEV_BSIZE, 0) < 0) { 21511288Smckusick perror("FS SEEK"); 21611288Smckusick exit(6); 21711288Smckusick } 21811288Smckusick if (write(fi, buf, size) != size) { 21911288Smckusick perror("FS WRITE"); 22011288Smckusick exit(7); 22111288Smckusick } 22211288Smckusick } 22311288Smckusick 22411288Smckusick bread(bno, buf, cnt) 22511288Smckusick daddr_t bno; 22611288Smckusick char *buf; 22711288Smckusick { 22811288Smckusick register i; 22911288Smckusick 23011288Smckusick if (lseek(fi, bno * DEV_BSIZE, 0) < 0) 23111288Smckusick return(1); 23211288Smckusick if ((i = read(fi, buf, cnt)) != cnt) { 23311288Smckusick for(i=0; i<sblock.fs_bsize; i++) 23411288Smckusick buf[i] = 0; 23511288Smckusick return (1); 23611288Smckusick } 23711288Smckusick return (0); 23811288Smckusick } 23911288Smckusick 24011288Smckusick /* VARARGS1 */ 24111288Smckusick fatal(fmt, arg1, arg2) 24211288Smckusick char *fmt, *arg1, *arg2; 24311288Smckusick { 24411288Smckusick 24511295Ssam fprintf(stderr, "tunefs: "); 24611288Smckusick fprintf(stderr, fmt, arg1, arg2); 24711288Smckusick putc('\n', stderr); 24811288Smckusick exit(10); 24911288Smckusick } 250