| /* |
| * Copyright 2002-2004 by Hans Reiser, licensing governed by |
| * reiserfsprogs/README |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| # include <config.h> |
| #endif |
| |
| #include "tune.h" |
| #include "misc/unaligned.h" |
| #include "util/misc.h" |
| #include "util/badblock.h" |
| #include "util/device.h" |
| #include "util/print.h" |
| |
| #include <stdarg.h> |
| #include <getopt.h> |
| #include <unistd.h> |
| #include <fcntl.h> |
| |
| #if 0 |
| #include <getopt.h> |
| #include <stdarg.h> |
| #include <string.h> |
| #include <errno.h> |
| #endif |
| |
| char *program_name; |
| |
| static void message( const char * fmt, ... ) |
| __attribute__ ((format (printf, 1, 2))); |
| |
| static void message( const char * fmt, ... ) |
| { |
| char buf[4096]; |
| va_list args; |
| |
| buf[0] = '\0'; |
| va_start( args, fmt ); |
| vsprintf( buf, fmt, args ); |
| va_end( args ); |
| |
| if( buf[0] != '\0' ) { |
| fprintf( stderr, "%s: %s\n", program_name, buf ); |
| } |
| } |
| |
| |
| reiserfs_filsys_t * fs; |
| |
| static void print_usage_and_exit(void) |
| { |
| message ("Usage: %s [options] device [block-count]\n" |
| "\n" |
| "Options:\n\n" |
| " -j | --journal-device file\tcurrent journal device\n" |
| " --journal-new-device file\tnew journal device\n" |
| " -o | --journal-new-offset N\tnew journal offset in blocks\n" |
| " -s | --journal-new-size N\tnew journal size in blocks\n" |
| " -t | --trans-max-size N\tnew journal max transaction size in blocks\n" |
| " --no-journal-available\tcurrent journal is not available\n" |
| " --make-journal-standard\tnew journal to be standard\n" |
| /*"\t-p | --keep-old-journal-param (keep parametrs from old journal to new one)\n"*/ |
| " -b | --add-badblocks file\tadd to bad block list\n" |
| " -B | --badblocks file\t\tset the bad block list\n" |
| " -u | --uuid UUID|random\tset new UUID\n" |
| " -l | --label LABEL\t\tset new label\n" |
| " -f | --force\t\t\tforce tuning, less confirmations\n" |
| " -V\t\t\t\tprint version and exit\n", program_name); |
| exit (1); |
| } |
| |
| unsigned long Journal_size = 0; |
| int Max_trans_size = JOURNAL_TRANS_MAX; |
| int Offset = 0; |
| __u16 Options = 0; |
| int Force = 0; |
| int Bads = 0; |
| char * LABEL; |
| unsigned char UUID[16]; |
| char * badblocks_file; |
| |
| /* If specified paramenters defines the standard journal, make it standard. */ |
| static int should_make_journal_standard (reiserfs_filsys_t * fs, char * j_new_dev_name) |
| { |
| if (!reiserfs_super_jr_magic (fs->fs_ondisk_sb)) |
| return 0; |
| /* |
| if (!util_user_confirmed (stderr, "ATTENTION! Filesystem with non-standard journal " |
| "found. Continue? (y/n):", "y\n")) { |
| exit(1); |
| } |
| */ |
| /* make sure journal is on main device, it has default size |
| and the file system has non-standard magic */ |
| |
| if (j_new_dev_name) { |
| /* new journal was specified - check if it is available */ |
| if (strcmp (j_new_dev_name, fs->fs_file_name)) |
| return 0; |
| |
| if (Journal_size && Journal_size != |
| reiserfs_journal_default(fs->fs_super_bh->b_blocknr, fs->fs_blocksize) + 1) |
| return 0; |
| |
| if (Max_trans_size && (Max_trans_size != JOURNAL_TRANS_MAX)) |
| return 0; |
| } else { |
| /* new journal was not specified - check ondisk journal params */ |
| |
| if (reiserfs_sb_get_reserved(fs->fs_ondisk_sb) < |
| reiserfs_journal_default (fs->fs_super_bh->b_blocknr, fs->fs_blocksize) + 1) |
| { |
| message ("Can not create standard journal of the size %llu", |
| reiserfs_journal_default(fs->fs_super_bh->b_blocknr, fs->fs_blocksize) + 1); |
| return 0; |
| } |
| } |
| |
| return 1; |
| } |
| |
| static int set_standard_journal_params (reiserfs_filsys_t * fs) |
| { |
| reiserfs_bh_t * bh; |
| |
| /* ondisk superblock update */ |
| |
| if (reiserfs_sb_get_version(fs->fs_ondisk_sb) == 0) |
| memcpy (fs->fs_ondisk_sb->s_v1.s_magic, REISERFS_3_5_SUPER_MAGIC_STRING, |
| strlen (REISERFS_3_5_SUPER_MAGIC_STRING)); |
| else if (reiserfs_sb_get_version(fs->fs_ondisk_sb) == 2) |
| memcpy (fs->fs_ondisk_sb->s_v1.s_magic, REISERFS_3_6_SUPER_MAGIC_STRING, |
| strlen (REISERFS_3_6_SUPER_MAGIC_STRING)); |
| else { |
| message ("Can not set standard reiserfs magic: unknown format found %u," |
| " try reiserfsck first", reiserfs_sb_get_version(fs->fs_ondisk_sb)); |
| return 0; |
| } |
| |
| reiserfs_jp_set_start (reiserfs_sb_jp(fs->fs_ondisk_sb), reiserfs_journal_start_must (fs)); |
| reiserfs_jp_set_dev (reiserfs_sb_jp(fs->fs_ondisk_sb), 0); |
| reiserfs_jp_set_size (reiserfs_sb_jp(fs->fs_ondisk_sb), reiserfs_journal_default(fs->fs_super_bh->b_blocknr, fs->fs_blocksize)); |
| |
| if (reiserfs_jp_get_tlen(reiserfs_sb_jp(fs->fs_ondisk_sb)) != JOURNAL_TRANS_MAX) |
| reiserfs_jp_set_tlen(reiserfs_sb_jp(fs->fs_ondisk_sb), JOURNAL_TRANS_MAX); |
| if (reiserfs_jp_get_max_batch(reiserfs_sb_jp(fs->fs_ondisk_sb)) != JOURNAL_MAX_BATCH) |
| reiserfs_jp_set_max_batch(reiserfs_sb_jp(fs->fs_ondisk_sb), JOURNAL_MAX_BATCH); |
| if (reiserfs_jp_get_commit_age(reiserfs_sb_jp(fs->fs_ondisk_sb)) != JOURNAL_MAX_COMMIT_AGE) |
| reiserfs_jp_set_commit_age(reiserfs_sb_jp(fs->fs_ondisk_sb), JOURNAL_MAX_COMMIT_AGE); |
| if (reiserfs_jp_get_trans_age(reiserfs_sb_jp(fs->fs_ondisk_sb)) != JOURNAL_MAX_TRANS_AGE) |
| reiserfs_jp_set_trans_age(reiserfs_sb_jp(fs->fs_ondisk_sb), JOURNAL_MAX_TRANS_AGE); |
| reiserfs_sb_set_reserved (fs->fs_ondisk_sb, 0); |
| |
| /* journal_header update */ |
| bh = reiserfs_buffer_open(fs->fs_journal_dev, |
| reiserfs_jp_get_start(reiserfs_sb_jp(fs->fs_ondisk_sb)) + |
| reiserfs_jp_get_size(reiserfs_sb_jp(fs->fs_ondisk_sb)), fs->fs_blocksize); |
| |
| if (!bh) { |
| message ("Cannot get the journal header block. reiserfs_buffer_open failed.\n"); |
| return 0; |
| } |
| ((struct reiserfs_journal_header *)(bh->b_data))->jh_journal = *(reiserfs_sb_jp(fs->fs_ondisk_sb)); |
| reiserfs_buffer_mkuptodate (bh, 1); |
| reiserfs_buffer_mkdirty (bh); |
| reiserfs_buffer_write(bh); |
| reiserfs_buffer_close(bh); |
| |
| return 1; |
| } |
| |
| void zero_journal (reiserfs_filsys_t * fs) |
| { |
| unsigned int i; |
| reiserfs_bh_t * bh; |
| unsigned long done; |
| unsigned long start, len; |
| |
| |
| fprintf (stderr, "Initializing journal - "); |
| |
| start = reiserfs_jp_get_start (reiserfs_sb_jp (fs->fs_ondisk_sb)); |
| len = reiserfs_jp_get_size (reiserfs_sb_jp (fs->fs_ondisk_sb)); |
| done = 0; |
| for (i = 0; i < len; i ++) { |
| util_misc_progress (stderr, &done, len, 1, 2); |
| bh = reiserfs_buffer_open (fs->fs_journal_dev, start + i, fs->fs_blocksize); |
| if (!bh) |
| misc_die ("zero_journal: reiserfs_buffer_open failed"); |
| memset (bh->b_data, 0, bh->b_size); |
| reiserfs_buffer_mkdirty (bh); |
| reiserfs_buffer_mkuptodate (bh, 1); |
| reiserfs_buffer_write (bh); |
| reiserfs_buffer_close (bh); |
| } |
| |
| fprintf (stderr, "\n"); |
| fflush (stderr); |
| } |
| |
| |
| static int str2int (char * str) |
| { |
| int val; |
| char * tmp; |
| |
| val = (int) strtol (str, &tmp, 0); |
| if (*tmp) |
| misc_die ("%s: strtol is unable to make an integer of %s\n", program_name, str); |
| return val; |
| } |
| |
| |
| static void set_transaction_max_size (char * str) |
| { |
| Max_trans_size = str2int( str ); |
| } |
| |
| |
| /* journal must fit into number of blocks pointed by first bitmap */ |
| static void set_journal_device_size (char * str) |
| { |
| Journal_size = str2int (str) ; |
| } |
| |
| |
| static void set_offset_in_journal_device (char * str) |
| { |
| Offset = str2int( str ); |
| } |
| |
| |
| static void callback_new_badblocks(reiserfs_filsys_t *fs, |
| reiserfs_path_t *badblock_path, |
| void *data) |
| { |
| reiserfs_ih_t *tmp_ih; |
| __u32 *ind_item, i; |
| |
| tmp_ih = REISERFS_PATH_IH(badblock_path); |
| ind_item = (__u32 *)REISERFS_PATH_ITEM(badblock_path); |
| |
| for (i = 0; i < reiserfs_ext_count(tmp_ih); i++) { |
| if (reiserfs_bitmap_test_bit(fs->fs_badblocks_bm, |
| d32_get (ind_item, i))) |
| { |
| message("Block %u is marked as bad already.", |
| d32_get (ind_item, i)); |
| |
| reiserfs_bitmap_clear_bit(fs->fs_badblocks_bm, |
| d32_get (ind_item, i)); |
| } |
| } |
| |
| reiserfs_tree_pathrelse (badblock_path); |
| } |
| |
| static void callback_clear_badblocks(reiserfs_filsys_t *fs, |
| reiserfs_path_t *badblock_path, |
| void *data) |
| { |
| reiserfs_ih_t *tmp_ih; |
| __u32 *ind_item, i; |
| |
| tmp_ih = REISERFS_PATH_IH(badblock_path); |
| ind_item = (__u32 *)REISERFS_PATH_ITEM(badblock_path); |
| |
| for (i = 0; i < reiserfs_ext_count(tmp_ih); i++) { |
| reiserfs_bitmap_clear_bit(fs->fs_bitmap2, |
| d32_get(ind_item, i)); |
| } |
| |
| reiserfs_tree_pathrelse (badblock_path); |
| } |
| |
| static void add_badblocks(reiserfs_filsys_t *fs) { |
| unsigned long i, marked = 0; |
| |
| if (reiserfs_bitmap_open (fs) < 0) { |
| message("Failed to open reiserfs ondisk bitmap.\n"); |
| reiserfs_fs_close(fs); |
| exit(1); |
| } |
| |
| if (util_badblock_load (fs, badblocks_file)) { |
| message("Failed to initialize the bad block bitmap.\n"); |
| reiserfs_fs_close(fs); |
| exit(1); |
| } |
| |
| if (Bads == 1) |
| reiserfs_badblock_traverse(fs, callback_new_badblocks, NULL); |
| else |
| reiserfs_badblock_traverse(fs, callback_clear_badblocks, NULL); |
| |
| for (i = 0; i < reiserfs_sb_get_blocks (fs->fs_ondisk_sb); i ++) { |
| if (reiserfs_bitmap_test_bit (fs->fs_badblocks_bm, i)) { |
| if (!reiserfs_bitmap_test_bit (fs->fs_bitmap2, i)) { |
| reiserfs_bitmap_set_bit (fs->fs_bitmap2, i); |
| marked++; |
| } else { |
| /* Check that this is a block */ |
| message("Bad block %lu is used already in reiserfs tree. " |
| "To mark it as a bad block use reiserfsck\n" |
| "--fix-fixable with -B option.", i); |
| |
| reiserfs_bitmap_clear_bit (fs->fs_badblocks_bm, i); |
| } |
| } |
| } |
| |
| if (marked) { |
| reiserfs_sb_set_free(fs->fs_ondisk_sb, reiserfs_sb_get_free(fs->fs_ondisk_sb) - |
| fs->fs_badblocks_bm->bm_set_bits); |
| reiserfs_buffer_mkdirty(fs->fs_super_bh); |
| } |
| |
| if (Bads == 1) { |
| /* fs->fs_badblocks_bm contains blocks which are not in the bad |
| block list yet. Merge it with what is in the tree already. */ |
| reiserfs_badblock_traverse(fs, reiserfs_badblock_extract, NULL); |
| } |
| |
| if (marked) { |
| reiserfs_badblock_flush(fs, 1); |
| } |
| |
| message("%lu blocks were marked as bad.", marked); |
| } |
| |
| int main (int argc, char **argv) |
| { |
| reiserfs_filsys_t * fs; |
| char * device_name; |
| char * jdevice_name; |
| char * j_new_device_name; |
| int c; |
| static int flag; |
| struct reiserfs_journal_header * j_head; |
| reiserfs_trans_t old, new; |
| int Is_journal_or_maxtrans_size_specified = 0; |
| |
| program_name = strrchr( argv[ 0 ], '/' ); |
| |
| if (program_name) |
| program_name++; |
| else |
| program_name = argv[ 0 ]; |
| |
| if (argc < 2) |
| print_usage_and_exit (); |
| |
| device_name = 0; |
| jdevice_name = 0; |
| j_new_device_name = 0; |
| |
| memset(UUID, 0, 16); |
| |
| while (1) { |
| static struct option options[] = { |
| {"journal-device", required_argument, 0, 'j'}, |
| {"journal-new-device", required_argument, &flag, OPT_NEW_J}, |
| {"journal-new-size", required_argument, 0, 's'}, |
| {"trans-max-size", required_argument, 0, 't'}, |
| {"journal-new-offset", required_argument, 0, 'o'}, |
| {"no-journal-available", no_argument, &flag, OPT_SKIP_J}, |
| /*{"keep-old-journal-param", no_argument, 0, 'p'},*/ |
| {"uuid", required_argument, 0, 'u'}, |
| {"label", required_argument, 0, 'l'}, |
| {"add-badblocks", required_argument, 0, 'b'}, |
| {"badblocks", required_argument, 0, 'B'}, |
| {"force", no_argument, 0, 'f'}, |
| {"make-journal-standard", no_argument, &flag, OPT_STANDARD}, |
| {0, 0, 0, 0} |
| }; |
| int option_index; |
| |
| c = getopt_long (argc, argv, "j:s:t:o:fu:l:b:B:V", |
| options, &option_index); |
| if (c == -1) |
| break; |
| |
| switch (c) { |
| case 0: |
| /* long-only optins */ |
| if (flag == OPT_NEW_J) { |
| Options |= OPT_NEW_J; |
| j_new_device_name = optarg; |
| } |
| if (flag == OPT_SKIP_J) { |
| Options |= OPT_SKIP_J; |
| } |
| if (flag == OPT_STANDARD) { |
| Options |=OPT_STANDARD; |
| } |
| break; |
| case 'j': /* --journal-device */ |
| jdevice_name = optarg; |
| break; |
| |
| case 's': /* --journal-new-size */ |
| set_journal_device_size (optarg); |
| Is_journal_or_maxtrans_size_specified = 1; |
| break; |
| |
| case 't': /* --trans-max-size */ |
| set_transaction_max_size (optarg); |
| Is_journal_or_maxtrans_size_specified = 1; |
| break; |
| |
| case 'o': /* --offset */ |
| set_offset_in_journal_device (optarg); |
| break; |
| |
| case 'f': |
| /* forces replacing standard journal with non-standard |
| one. Specified more than once - allows to avoid asking for |
| confirmation */ |
| Force ++; |
| break; |
| case 'b': /* --add-badblocks */ |
| badblocks_file = optarg; |
| Bads = 1; |
| break; |
| case 'B': /* --badblocks */ |
| badblocks_file = optarg; |
| Bads = 2; |
| break; |
| case 'u': |
| /* UUID */ |
| #if defined(HAVE_LIBUUID) && defined(HAVE_UUID_UUID_H) |
| if (!strcmp(optarg, "random")) { |
| uuid_generate(UUID); |
| } else { |
| if (uuid_parse(optarg, UUID) < 0) { |
| message ("Invalid UUID '%s' was specified\n", optarg); |
| return 1; |
| } |
| } |
| #else |
| message ("Cannot set the UUID, uuidlib was not found " |
| "by configure.\n"); |
| return 1; |
| #endif |
| break; |
| case 'l': |
| /* LABEL */ |
| LABEL = optarg; |
| break; |
| case 'V': |
| util_print_banner("reiserfstune"); |
| exit(0); |
| #if 0 |
| case 'J': /* --journal-new-device */ |
| Options |= OPT_NEW_J; |
| j_new_device_name = optarg; |
| break; |
| |
| case 'u': /* --no-journal-available */ |
| Options |= OPT_SKIPJ; |
| break; |
| |
| case 'p': /* --keep-old-journal-param */ |
| Options |= OPT_KEEPO; |
| break; |
| #endif |
| default: |
| print_usage_and_exit(); |
| } |
| } |
| |
| if (optind != argc - 1) |
| print_usage_and_exit (); |
| |
| /* device to be formatted */ |
| device_name = argv [optind]; |
| |
| fs = reiserfs_fs_open (device_name, O_RDONLY, 0, NULL, 1); |
| if (fs == NULL) { |
| message ("Cannot open reiserfs on %s", device_name); |
| return 1; |
| } |
| |
| /* journal was opened or it wasn't opened but the option |
| --no-journal-available has been specified by user */ |
| |
| /* make sure filesystem is not mounted */ |
| if (util_device_mounted(fs->fs_file_name) > 0) { |
| /* fixme: it can not be mounted, btw */ |
| message ("Reiserfstune is not allowed to be run on mounted filesystem."); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| |
| if (!reiserfs_sb_state_ok (fs)) { |
| message ("Filesystem looks not cleanly umounted, check the consistency first.\n"); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| |
| reiserfs_fs_reopen (fs, O_RDWR); |
| |
| if (badblocks_file) { |
| add_badblocks(fs); |
| |
| reiserfs_fs_close(fs); |
| exit(0); |
| } |
| |
| if (!jdevice_name && !(Options & OPT_SKIP_J)) { |
| message ("Journal device has not been specified. Assuming journal is on the main " |
| "device (%s).\n", device_name); |
| jdevice_name = device_name; |
| } |
| |
| if (jdevice_name && (Options & OPT_SKIP_J)) { |
| message ("Either specify journal device, " |
| "or choose the option --no-journal-available"); |
| return 1; |
| } |
| |
| if (j_new_device_name && (Options & OPT_STANDARD)) { |
| /* New device was specified and --make-journal-standard was also. */ |
| message ("Either specify new journal device, " |
| "or choose the option --make-journal-standard"); |
| return 1; |
| } |
| |
| /* now we try to open journal, it makes sence if there is no the flag |
| NEED_TUNE in ondisk superblock and --no-journal available is not |
| specified. */ |
| if (reiserfs_jp_get_magic(reiserfs_sb_jp(fs->fs_ondisk_sb)) != NEED_TUNE && |
| !(Options & OPT_SKIP_J)) |
| { |
| if (reiserfs_journal_open (fs, jdevice_name, O_RDWR |
| #if defined(O_LARGEFILE) |
| | O_LARGEFILE |
| #endif |
| )) |
| { |
| message ("Failed to open the journal device (%s).", jdevice_name); |
| return 1; |
| } |
| |
| if (reiserfs_journal_params_check(fs)) { |
| message ("Unable to open old journal. Wrong journal parameters."); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| } |
| |
| /* in spite of journal was opened, the file system can be non-consistent or |
| there are non-replayed transaction in journal, |
| make sure it isn't (if there is no the flag NEED_TUNE in ondisk superblock */ |
| if (reiserfs_jp_get_magic(reiserfs_sb_jp(fs->fs_ondisk_sb)) != NEED_TUNE && |
| reiserfs_journal_opened (fs)) { |
| j_head = (struct reiserfs_journal_header *)(fs->fs_jh_bh->b_data); |
| if (reiserfs_journal_get_transactions(fs, &old, &new)) { |
| if (new.trans_id != reiserfs_jh_get_flushed(j_head)) { |
| message ("There are non-replayed transaction in old journal," |
| " check filesystem consistency first"); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| } |
| if (!reiserfs_sb_state_ok (fs)) { |
| message ("Check filesystem consistency first"); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| } |
| |
| /* set UUID and LABEL if specified */ |
| if (fs->fs_format == REISERFS_FORMAT_3_6) { |
| #if defined(HAVE_LIBUUID) && defined(HAVE_UUID_UUID_H) |
| if (!uuid_is_null(UUID)) { |
| memcpy (fs->fs_ondisk_sb->s_uuid, UUID, 16); |
| reiserfs_buffer_mkdirty (fs->fs_super_bh); |
| fs->fs_dirt = 1; |
| } |
| #endif |
| if (LABEL != NULL) { |
| if (strlen (LABEL) > 16) |
| message ("Specified LABEL is longer then 16 characters, will be truncated\n"); |
| strncpy ((char *)fs->fs_ondisk_sb->s_label, LABEL, 16); |
| reiserfs_buffer_mkdirty (fs->fs_super_bh); |
| fs->fs_dirt = 1; |
| } |
| } else { |
| #if defined(HAVE_LIBUUID) && defined(HAVE_UUID_UUID_H) |
| if (!uuid_is_null(UUID)) |
| reiserfs_exit (1, "UUID cannot be specified for 3.5 format\n"); |
| #endif |
| if (LABEL) |
| reiserfs_exit (1, "LABEL cannot be specified for 3.5 format\n"); |
| } |
| |
| if (!j_new_device_name) { |
| |
| /* new journal device hasn't been specified */ |
| printf ("Current parameters:\n"); |
| reiserfs_super_print_state (stdout, fs); |
| reiserfs_node_print (stdout, fs, fs->fs_super_bh); |
| |
| if ((Options & OPT_STANDARD) |
| && should_make_journal_standard(fs, j_new_device_name)) |
| { |
| if (!util_user_confirmed (stderr, "ATTENTION! Filesystem with " |
| "non-standard journal found. Continue? (y/n):", "y\n")) |
| { |
| exit(1); |
| } |
| |
| fs->fs_journal_dev = fs->fs_dev; |
| if (set_standard_journal_params (fs)) { |
| printf ("\nNew parameters:\n"); |
| reiserfs_super_print_state (stdout, fs); |
| reiserfs_node_print (stdout, fs, fs->fs_super_bh); |
| printf ("New journal parameters:\n"); |
| reiserfs_journal_print_params (stdout, reiserfs_sb_jp (fs->fs_ondisk_sb)); |
| reiserfs_buffer_mkdirty (fs->fs_super_bh); |
| reiserfs_buffer_mkuptodate (fs->fs_super_bh, 1); |
| reiserfs_fs_close (fs); |
| printf ("Syncing.."); fflush (stdout); |
| sync (); |
| printf ("ok\n"); |
| return 0; |
| } |
| } |
| |
| if (Is_journal_or_maxtrans_size_specified) { |
| /* new journal device hasn't been specified, but |
| journal size or max transaction size have been, so we suppose |
| that journal device remains the same */ |
| if (!reiserfs_journal_opened (fs)) { |
| message("Cannot set up new paramenters for not specified journal."); |
| return 1; |
| } |
| |
| j_new_device_name = jdevice_name; |
| } else { |
| /* the only parameter has been specified is device_name, so |
| there is nothing to do */ |
| reiserfs_fs_close (fs); |
| return 0; |
| } |
| } |
| |
| /* new journal device has been specified */ |
| /* make sure new journal device is block device file */ |
| if (!util_device_formatable (j_new_device_name, Force)) { |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| |
| if (!strcmp (device_name, j_new_device_name)) { |
| unsigned long reserved, journal_size; |
| /* we have to put journal on main device. It is only possible if there |
| is enough space reserved by mkreiserfs */ |
| |
| if (!reiserfs_super_jr_magic (fs->fs_ondisk_sb)) |
| /* standard journal */ |
| reserved = reiserfs_jp_get_size(reiserfs_sb_jp(fs->fs_ondisk_sb)) + 1; |
| else |
| /* non-standard journal */ |
| reserved = reiserfs_sb_get_reserved (fs->fs_ondisk_sb); |
| |
| journal_size = Journal_size; |
| |
| if (!journal_size) { |
| journal_size = reiserfs_journal_default(fs->fs_super_bh->b_blocknr, fs->fs_blocksize) + 1; |
| message("Journal size has not been specified. Assuming it is the default size (%lu)", |
| journal_size); |
| } |
| |
| if (journal_size + Offset > reiserfs_journal_start_must (fs) + reserved) { |
| message ("There is no enough space reserved for journal on main " |
| "device (journal_size=%lu, reserved=%lu)\n", journal_size, |
| reserved); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| } |
| |
| message ("Current journal parameters:"); |
| reiserfs_journal_print_params (stdout, reiserfs_sb_jp (fs->fs_ondisk_sb)); |
| |
| if (!reiserfs_super_jr_magic (fs->fs_ondisk_sb)) { |
| /* we have standard journal, so check if we can convert it |
| to non-standard one */ |
| |
| /* |
| if (!should_make_journal_non_standard (Force)) { |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| */ |
| |
| if (reiserfs_super_36_magic (fs->fs_ondisk_sb)) |
| reiserfs_sb_set_version (fs->fs_ondisk_sb, REISERFS_FORMAT_3_6); |
| else if (reiserfs_super_35_magic (fs->fs_ondisk_sb)) |
| reiserfs_sb_set_version (fs->fs_ondisk_sb, REISERFS_FORMAT_3_5); |
| else { |
| message ("Could not convert from unknown version, try reiserfsck first"); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| |
| memcpy (fs->fs_ondisk_sb->s_v1.s_magic, REISERFS_JR_SUPER_MAGIC_STRING, |
| strlen (REISERFS_JR_SUPER_MAGIC_STRING)); |
| reiserfs_sb_set_reserved (fs->fs_ondisk_sb, |
| reiserfs_jp_get_size (reiserfs_sb_jp(fs->fs_ondisk_sb)) + 1); |
| } |
| |
| /* now we are going to close old journal and to create a new one */ |
| reiserfs_journal_close (fs); |
| |
| if (!reiserfs_journal_create (fs, j_new_device_name, Offset, |
| Journal_size, Max_trans_size)) { |
| message ("Could not create new journal"); |
| reiserfs_fs_close (fs); |
| return 1; |
| } |
| |
| if (should_make_journal_standard (fs, j_new_device_name)) |
| set_standard_journal_params (fs); |
| |
| message ("New journal parameters:"); |
| reiserfs_journal_print_params (stdout, reiserfs_sb_jp (fs->fs_ondisk_sb)); |
| |
| reiserfs_node_print (stdout, fs, fs->fs_super_bh); |
| |
| if (Force < 2) { |
| message ("ATTENTION: YOU ARE ABOUT TO SETUP THE NEW JOURNAL FOR THE \"%s\"!\n" |
| "AREA OF \"%s\" DEDICATED FOR JOURNAL WILL BE ZEROED!", |
| device_name, j_new_device_name); |
| |
| if (!util_user_confirmed (stderr, "Continue (y/n):", "y\n")) { |
| return 1; |
| } |
| } |
| |
| zero_journal (fs); |
| reiserfs_fs_close (fs); |
| |
| printf ("Syncing.."); fflush (stdout); |
| sync (); |
| printf ("ok\n"); |
| |
| return 0; |
| } |