# include <alloca.h>
#endif
+#define WIMLIB_COMPRESSION_TYPE_INVALID (-1)
+
#ifdef __WIN32__
# include "imagex-win32.h"
# define print_security_descriptor win32_print_security_descriptor
CMD_UNMOUNT,
#endif
CMD_UPDATE,
+ CMD_VERIFY,
CMD_MAX,
};
IMAGEX_NO_GLOBS_OPTION,
IMAGEX_NULLGLOB_OPTION,
IMAGEX_ONE_FILE_ONLY_OPTION,
- IMAGEX_PACK_CHUNK_SIZE_OPTION,
- IMAGEX_PACK_COMPRESS_OPTION,
- IMAGEX_PACK_STREAMS_OPTION,
IMAGEX_PATH_OPTION,
IMAGEX_PIPABLE_OPTION,
IMAGEX_PRESERVE_DIR_STRUCTURE_OPTION,
IMAGEX_RESUME_OPTION,
IMAGEX_RPFIX_OPTION,
IMAGEX_SOFT_OPTION,
+ IMAGEX_SOLID_OPTION,
+ IMAGEX_SOLID_CHUNK_SIZE_OPTION,
+ IMAGEX_SOLID_COMPRESS_OPTION,
IMAGEX_SOURCE_LIST_OPTION,
IMAGEX_STAGING_DIR_OPTION,
IMAGEX_STREAMS_INTERFACE_OPTION,
{T("compress"), required_argument, NULL, IMAGEX_COMPRESS_OPTION},
{T("compress-slow"), no_argument, NULL, IMAGEX_COMPRESS_SLOW_OPTION},
{T("chunk-size"), required_argument, NULL, IMAGEX_CHUNK_SIZE_OPTION},
- {T("pack-chunk-size"), required_argument, NULL, IMAGEX_PACK_CHUNK_SIZE_OPTION},
- {T("solid-chunk-size"),required_argument, NULL, IMAGEX_PACK_CHUNK_SIZE_OPTION},
- {T("pack-compress"), required_argument, NULL, IMAGEX_PACK_COMPRESS_OPTION},
- {T("solid-compress"),required_argument, NULL, IMAGEX_PACK_COMPRESS_OPTION},
- {T("pack-streams"), no_argument, NULL, IMAGEX_PACK_STREAMS_OPTION},
- {T("solid"), no_argument, NULL, IMAGEX_PACK_STREAMS_OPTION},
+ {T("solid"), no_argument, NULL, IMAGEX_SOLID_OPTION},
+ {T("pack-streams"), no_argument, NULL, IMAGEX_SOLID_OPTION},
+ {T("solid-compress"),required_argument, NULL, IMAGEX_SOLID_COMPRESS_OPTION},
+ {T("pack-compress"), required_argument, NULL, IMAGEX_SOLID_COMPRESS_OPTION},
+ {T("solid-chunk-size"),required_argument, NULL, IMAGEX_SOLID_CHUNK_SIZE_OPTION},
+ {T("pack-chunk-size"), required_argument, NULL, IMAGEX_SOLID_CHUNK_SIZE_OPTION},
{T("config"), required_argument, NULL, IMAGEX_CONFIG_OPTION},
{T("dereference"), no_argument, NULL, IMAGEX_DEREFERENCE_OPTION},
{T("flags"), required_argument, NULL, IMAGEX_FLAGS_OPTION},
{T("nocheck"), no_argument, NULL, IMAGEX_NOCHECK_OPTION},
{T("no-check"), no_argument, NULL, IMAGEX_NOCHECK_OPTION},
{T("compress"), required_argument, NULL, IMAGEX_COMPRESS_OPTION},
+ {T("recompress"), no_argument, NULL, IMAGEX_RECOMPRESS_OPTION},
{T("compress-slow"), no_argument, NULL, IMAGEX_COMPRESS_SLOW_OPTION},
- {T("pack-streams"),no_argument, NULL, IMAGEX_PACK_STREAMS_OPTION},
- {T("solid"), no_argument, NULL, IMAGEX_PACK_STREAMS_OPTION},
{T("chunk-size"), required_argument, NULL, IMAGEX_CHUNK_SIZE_OPTION},
- {T("pack-chunk-size"), required_argument, NULL, IMAGEX_PACK_CHUNK_SIZE_OPTION},
- {T("solid-chunk-size"),required_argument, NULL, IMAGEX_PACK_CHUNK_SIZE_OPTION},
- {T("pack-compress"), required_argument, NULL, IMAGEX_PACK_COMPRESS_OPTION},
- {T("solid-compress"),required_argument, NULL, IMAGEX_PACK_COMPRESS_OPTION},
+ {T("solid"), no_argument, NULL, IMAGEX_SOLID_OPTION},
+ {T("pack-streams"),no_argument, NULL, IMAGEX_SOLID_OPTION},
+ {T("solid-compress"),required_argument, NULL, IMAGEX_SOLID_COMPRESS_OPTION},
+ {T("pack-compress"), required_argument, NULL, IMAGEX_SOLID_COMPRESS_OPTION},
+ {T("solid-chunk-size"),required_argument, NULL, IMAGEX_SOLID_CHUNK_SIZE_OPTION},
+ {T("pack-chunk-size"), required_argument, NULL, IMAGEX_SOLID_CHUNK_SIZE_OPTION},
{T("ref"), required_argument, NULL, IMAGEX_REF_OPTION},
{T("threads"), required_argument, NULL, IMAGEX_THREADS_OPTION},
{T("rebuild"), no_argument, NULL, IMAGEX_REBUILD_OPTION},
{T("compress"), required_argument, NULL, IMAGEX_COMPRESS_OPTION},
{T("recompress"), no_argument, NULL, IMAGEX_RECOMPRESS_OPTION},
{T("compress-slow"), no_argument, NULL, IMAGEX_COMPRESS_SLOW_OPTION},
- {T("recompress-slow"), no_argument, NULL, IMAGEX_COMPRESS_SLOW_OPTION},
+ {T("recompress-slow"), no_argument, NULL, IMAGEX_COMPRESS_SLOW_OPTION},
{T("chunk-size"), required_argument, NULL, IMAGEX_CHUNK_SIZE_OPTION},
- {T("pack-chunk-size"), required_argument, NULL, IMAGEX_PACK_CHUNK_SIZE_OPTION},
- {T("solid-chunk-size"),required_argument, NULL, IMAGEX_PACK_CHUNK_SIZE_OPTION},
- {T("pack-compress"), required_argument, NULL, IMAGEX_PACK_COMPRESS_OPTION},
- {T("solid-compress"),required_argument, NULL, IMAGEX_PACK_COMPRESS_OPTION},
- {T("pack-streams"),no_argument, NULL, IMAGEX_PACK_STREAMS_OPTION},
- {T("solid"), no_argument, NULL, IMAGEX_PACK_STREAMS_OPTION},
+ {T("solid"), no_argument, NULL, IMAGEX_SOLID_OPTION},
+ {T("pack-streams"),no_argument, NULL, IMAGEX_SOLID_OPTION},
+ {T("solid-compress"),required_argument, NULL, IMAGEX_SOLID_COMPRESS_OPTION},
+ {T("pack-compress"), required_argument, NULL, IMAGEX_SOLID_COMPRESS_OPTION},
+ {T("solid-chunk-size"),required_argument, NULL, IMAGEX_SOLID_CHUNK_SIZE_OPTION},
+ {T("pack-chunk-size"), required_argument, NULL, IMAGEX_SOLID_CHUNK_SIZE_OPTION},
{T("threads"), required_argument, NULL, IMAGEX_THREADS_OPTION},
{T("pipable"), no_argument, NULL, IMAGEX_PIPABLE_OPTION},
{T("not-pipable"), no_argument, NULL, IMAGEX_NOT_PIPABLE_OPTION},
{NULL, 0, NULL, 0},
};
+static const struct option verify_options[] = {
+ {T("ref"), required_argument, NULL, IMAGEX_REF_OPTION},
+ {T("nocheck"), no_argument, NULL, IMAGEX_NOCHECK_OPTION},
+
+ {NULL, 0, NULL, 0},
+};
+
#if 0
# define _format_attribute(type, format_str, args_start) \
__attribute__((format(type, format_str, args_start)))
return ret;
}
+static void
+print_available_compression_types(FILE *fp)
+{
+ static const tchar *s =
+ T(
+ "Available compression types:\n"
+ "\n"
+ " none\n"
+ " xpress (alias: \"fast\")\n"
+ " lzx (alias: \"maximum\") (default for capture)\n"
+ " lzms (alias: \"recovery\")\n"
+ "\n"
+ );
+ tfputs(s, fp);
+}
+
/* Parse the argument to --compress */
static int
-get_compression_type(const tchar *optarg)
+get_compression_type(tchar *optarg)
{
+ int ctype;
+ unsigned int compression_level = 0;
+ tchar *plevel;
+
+ plevel = tstrchr(optarg, T(':'));
+ if (plevel) {
+ tchar *ptmp;
+ unsigned long ultmp;
+
+ *plevel++ = T('\0');
+ ultmp = tstrtoul(plevel, &ptmp, 10);
+ if (ultmp >= UINT_MAX || ultmp == 0 || *ptmp || ptmp == plevel) {
+ imagex_error(T("Compression level must be a positive integer! "
+ "e.g. --compress=lzx:80"));
+ return WIMLIB_COMPRESSION_TYPE_INVALID;
+ }
+ compression_level = ultmp;
+ }
+
if (!tstrcasecmp(optarg, T("maximum")) ||
!tstrcasecmp(optarg, T("lzx")) ||
!tstrcasecmp(optarg, T("max")))
- return WIMLIB_COMPRESSION_TYPE_LZX;
+ ctype = WIMLIB_COMPRESSION_TYPE_LZX;
else if (!tstrcasecmp(optarg, T("fast")) || !tstrcasecmp(optarg, T("xpress")))
- return WIMLIB_COMPRESSION_TYPE_XPRESS;
+ ctype = WIMLIB_COMPRESSION_TYPE_XPRESS;
else if (!tstrcasecmp(optarg, T("recovery")) || !tstrcasecmp(optarg, T("lzms")))
- return WIMLIB_COMPRESSION_TYPE_LZMS;
+ ctype = WIMLIB_COMPRESSION_TYPE_LZMS;
else if (!tstrcasecmp(optarg, T("none")))
- return WIMLIB_COMPRESSION_TYPE_NONE;
+ ctype = WIMLIB_COMPRESSION_TYPE_NONE;
else {
- imagex_error(T("Invalid compression type \"%"TS"\"! Must be "
- "\"maximum\", \"fast\", or \"none\"."), optarg);
+ imagex_error(T("Invalid compression type \"%"TS"\"!"), optarg);
+ print_available_compression_types(stderr);
return WIMLIB_COMPRESSION_TYPE_INVALID;
}
+
+ if (compression_level != 0)
+ wimlib_set_default_compression_level(ctype, compression_level);
+ return ctype;
}
static void
set_compress_slow(void)
{
- static const struct wimlib_lzx_compressor_params lzx_slow_params = {
- .hdr = {
- .size = sizeof(struct wimlib_lzx_compressor_params),
- },
- .algorithm = WIMLIB_LZX_ALGORITHM_SLOW,
- .alg_params = {
- .slow = {
- .use_len2_matches = 1,
- .nice_match_length = 96,
- .num_optim_passes = 4,
- .max_search_depth = 100,
- .main_nostat_cost = 15,
- .len_nostat_cost = 15,
- .aligned_nostat_cost = 7,
- },
- },
- };
-
- static const struct wimlib_lzms_compressor_params lzms_slow_params = {
- .hdr = {
- .size = sizeof(struct wimlib_lzms_compressor_params),
- },
- .min_match_length = 2,
- .max_match_length = UINT32_MAX,
- .nice_match_length = 96,
- .max_search_depth = 100,
- .optim_array_length = 1024,
- };
-
- wimlib_set_default_compressor_params(WIMLIB_COMPRESSION_TYPE_LZX,
- &lzx_slow_params.hdr);
-
- wimlib_set_default_compressor_params(WIMLIB_COMPRESSION_TYPE_LZMS,
- &lzms_slow_params.hdr);
+#if 0
+ fprintf(stderr, "WARNING: the '--compress-slow' option is deprecated.\n"
+ " Use the '--compress=TYPE:LEVEL' option instead.\n");
+#endif
+ wimlib_set_default_compression_level(-1, 100);
}
struct string_set {
imagex_printf(T("\nWARNING: Excluding unsupported file or directory\n"
" \"%"TS"\" from capture\n"), info->scan.cur_path);
break;
- case WIMLIB_SCAN_DENTRY_EXCLUDED_SYMLINK:
- imagex_printf(T("\nWARNING: Ignoring absolute symbolic link "
- "with out-of-tree target:\n"
- " \"%"TS"\" => \"%"TS"\"\n"
- " (Use --norpfix to capture "
- "absolute symbolic links as-is)\n"),
- info->scan.cur_path, info->scan.symlink_target);
- break;
case WIMLIB_SCAN_DENTRY_FIXED_SYMLINK:
/* Symlink fixups are enabled by default. This is
* mainly intended for Windows, which for some reason
* uses absolute junctions (with drive letters!) in the
* default installation. On UNIX-like systems, warn the
* user when fixing the target of an absolute symbolic
- * link, so they know to disable this if they want.
- * (Although, more likely they will get the warning
- * about an absolute symbolic link with an out-of-tree
- * target first.) */
+ * link, so they know to disable this if they want. */
#ifndef __WIN32__
imagex_printf(T("\nWARNING: Adjusted target of "
"absolute symbolic link \"%"TS"\"\n"
info->scan.cur_path);
#endif
break;
+ default:
+ break;
}
break;
case WIMLIB_PROGRESS_MSG_SCAN_END:
T("NTFS volume") : T("directory")),
info->extract.target);
break;
+ case WIMLIB_PROGRESS_MSG_EXTRACT_FILE_STRUCTURE:
+ if (info->extract.end_file_count >= 2000) {
+ percent_done = TO_PERCENT(info->extract.current_file_count,
+ info->extract.end_file_count);
+ imagex_printf(T("\rCreating files: %"PRIu64" of %"PRIu64" (%u%%) done"),
+ info->extract.current_file_count,
+ info->extract.end_file_count, percent_done);
+ if (info->extract.current_file_count == info->extract.end_file_count)
+ imagex_printf(T("\n"));
+ }
+ break;
case WIMLIB_PROGRESS_MSG_EXTRACT_STREAMS:
percent_done = TO_PERCENT(info->extract.completed_bytes,
info->extract.total_bytes);
if (info->extract.completed_bytes >= info->extract.total_bytes)
imagex_printf(T("\n"));
break;
+ case WIMLIB_PROGRESS_MSG_EXTRACT_METADATA:
+ if (info->extract.end_file_count >= 2000) {
+ percent_done = TO_PERCENT(info->extract.current_file_count,
+ info->extract.end_file_count);
+ imagex_printf(T("\rApplying metadata to files: %"PRIu64" of %"PRIu64" (%u%%) done"),
+ info->extract.current_file_count,
+ info->extract.end_file_count, percent_done);
+ if (info->extract.current_file_count == info->extract.end_file_count)
+ imagex_printf(T("\n"));
+ }
+ break;
case WIMLIB_PROGRESS_MSG_EXTRACT_SPWM_PART_BEGIN:
if (info->extract.total_parts != 1) {
imagex_printf(T("\nReading split pipable WIM part %u of %u\n"),
}
}
break;
+ case WIMLIB_PROGRESS_MSG_BEGIN_VERIFY_IMAGE:
+ imagex_printf(T("Verifying metadata for image %"PRIu32" of %"PRIu32"\n"),
+ info->verify_image.current_image,
+ info->verify_image.total_images);
+ break;
+ case WIMLIB_PROGRESS_MSG_VERIFY_STREAMS:
+ percent_done = TO_PERCENT(info->verify_streams.completed_bytes,
+ info->verify_streams.total_bytes);
+ unit_shift = get_unit(info->verify_streams.total_bytes, &unit_name);
+ imagex_printf(T("\rVerifying streams: "
+ "%"PRIu64" %"TS" of %"PRIu64" %"TS" (%u%%) done"),
+ info->verify_streams.completed_bytes >> unit_shift,
+ unit_name,
+ info->verify_streams.total_bytes >> unit_shift,
+ unit_name,
+ percent_done);
+ if (info->verify_streams.completed_bytes == info->verify_streams.total_bytes)
+ imagex_printf(T("\n"));
+ break;
default:
break;
}
int write_flags = 0;
int compression_type = WIMLIB_COMPRESSION_TYPE_INVALID;
uint32_t chunk_size = UINT32_MAX;
- uint32_t pack_chunk_size = UINT32_MAX;
- int pack_ctype = WIMLIB_COMPRESSION_TYPE_INVALID;
+ uint32_t solid_chunk_size = UINT32_MAX;
+ int solid_ctype = WIMLIB_COMPRESSION_TYPE_INVALID;
const tchar *wimfile;
int wim_fd;
const tchar *name;
struct wimlib_capture_source *capture_sources;
size_t num_sources;
bool name_defaulted;
- bool compress_slow = false;
for_opt(c, capture_or_append_options) {
switch (c) {
goto out_err;
break;
case IMAGEX_COMPRESS_SLOW_OPTION:
- compress_slow = true;
+ set_compress_slow();
break;
case IMAGEX_CHUNK_SIZE_OPTION:
chunk_size = parse_chunk_size(optarg);
if (chunk_size == UINT32_MAX)
goto out_err;
break;
- case IMAGEX_PACK_CHUNK_SIZE_OPTION:
- pack_chunk_size = parse_chunk_size(optarg);
- if (pack_chunk_size == UINT32_MAX)
+ case IMAGEX_SOLID_CHUNK_SIZE_OPTION:
+ solid_chunk_size = parse_chunk_size(optarg);
+ if (solid_chunk_size == UINT32_MAX)
goto out_err;
break;
- case IMAGEX_PACK_COMPRESS_OPTION:
- pack_ctype = get_compression_type(optarg);
- if (pack_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
+ case IMAGEX_SOLID_COMPRESS_OPTION:
+ solid_ctype = get_compression_type(optarg);
+ if (solid_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
- case IMAGEX_PACK_STREAMS_OPTION:
+ case IMAGEX_SOLID_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_PACK_STREAMS;
break;
case IMAGEX_FLAGS_OPTION:
/* With --wimboot, default to XPRESS compression. */
compression_type = WIMLIB_COMPRESSION_TYPE_XPRESS;
} else if (write_flags & WIMLIB_WRITE_FLAG_PACK_STREAMS) {
- /* With --pack-streams or --solid, default to LZMS
- * compression. (However, this will not affect packed
- * resources!) */
+ /* With --solid, default to LZMS compression. (However,
+ * this will not affect solid blocks!) */
compression_type = WIMLIB_COMPRESSION_TYPE_LZMS;
} else {
- /* Otherwise, default to LZX compression in fast mode.
- */
+ /* Otherwise, default to LZX compression. */
compression_type = WIMLIB_COMPRESSION_TYPE_LZX;
- if (!compress_slow && pack_ctype != WIMLIB_COMPRESSION_TYPE_LZX) {
- struct wimlib_lzx_compressor_params params = {
- .hdr.size = sizeof(params),
- .algorithm = WIMLIB_LZX_ALGORITHM_FAST,
- .use_defaults = 1,
- };
- wimlib_set_default_compressor_params(WIMLIB_COMPRESSION_TYPE_LZX,
- ¶ms.hdr);
- }
}
}
- if (compress_slow)
- set_compress_slow();
-
if (!tstrcmp(wimfile, T("-"))) {
/* Writing captured WIM to standard output. */
#if 0
if (ret)
goto out_free_wim;
}
- if (pack_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
- ret = wimlib_set_output_pack_compression_type(wim, pack_ctype);
+ if (solid_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
+ ret = wimlib_set_output_pack_compression_type(wim, solid_ctype);
if (ret)
goto out_free_wim;
}
- if (pack_chunk_size != UINT32_MAX) {
- ret = wimlib_set_output_pack_chunk_size(wim, pack_chunk_size);
+ if (solid_chunk_size != UINT32_MAX) {
+ ret = wimlib_set_output_pack_chunk_size(wim, solid_chunk_size);
if (ret)
goto out_free_wim;
}
STRING_SET(refglobs);
unsigned num_threads = 0;
uint32_t chunk_size = UINT32_MAX;
- uint32_t pack_chunk_size = UINT32_MAX;
- int pack_ctype = WIMLIB_COMPRESSION_TYPE_INVALID;
+ uint32_t solid_chunk_size = UINT32_MAX;
+ int solid_ctype = WIMLIB_COMPRESSION_TYPE_INVALID;
for_opt(c, export_options) {
switch (c) {
goto out_err;
break;
case IMAGEX_COMPRESS_SLOW_OPTION:
- write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
set_compress_slow();
+ write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
break;
- case IMAGEX_PACK_STREAMS_OPTION:
+ case IMAGEX_RECOMPRESS_OPTION:
+ write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
+ break;
+ case IMAGEX_SOLID_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_PACK_STREAMS;
break;
case IMAGEX_CHUNK_SIZE_OPTION:
if (chunk_size == UINT32_MAX)
goto out_err;
break;
- case IMAGEX_PACK_CHUNK_SIZE_OPTION:
- pack_chunk_size = parse_chunk_size(optarg);
- if (pack_chunk_size == UINT32_MAX)
+ case IMAGEX_SOLID_CHUNK_SIZE_OPTION:
+ solid_chunk_size = parse_chunk_size(optarg);
+ if (solid_chunk_size == UINT32_MAX)
goto out_err;
break;
- case IMAGEX_PACK_COMPRESS_OPTION:
- pack_ctype = get_compression_type(optarg);
- if (pack_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
+ case IMAGEX_SOLID_COMPRESS_OPTION:
+ solid_ctype = get_compression_type(optarg);
+ if (solid_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
case IMAGEX_REF_OPTION:
if (compression_type == WIMLIB_COMPRESSION_TYPE_INVALID) {
/* The user did not specify a compression type; default
- * to that of the source WIM, unless --pack-streams,
- * --solid, or --wimboot was specified. */
+ * to that of the source WIM, unless --solid or
+ * --wimboot was specified. */
if (write_flags & WIMLIB_WRITE_FLAG_PACK_STREAMS)
compression_type = WIMLIB_COMPRESSION_TYPE_LZMS;
if (ret)
goto out_free_dest_wim;
}
- if (pack_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
- ret = wimlib_set_output_pack_compression_type(dest_wim, pack_ctype);
+ if (solid_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
+ ret = wimlib_set_output_pack_compression_type(dest_wim, solid_ctype);
if (ret)
goto out_free_dest_wim;
}
- if (pack_chunk_size != UINT32_MAX) {
- ret = wimlib_set_output_pack_chunk_size(dest_wim, pack_chunk_size);
+ if (solid_chunk_size != UINT32_MAX) {
+ ret = wimlib_set_output_pack_chunk_size(dest_wim, solid_chunk_size);
if (ret)
goto out_free_dest_wim;
}
int write_flags = WIMLIB_WRITE_FLAG_REBUILD;
int compression_type = WIMLIB_COMPRESSION_TYPE_INVALID;
uint32_t chunk_size = UINT32_MAX;
- uint32_t pack_chunk_size = UINT32_MAX;
- int pack_ctype = WIMLIB_COMPRESSION_TYPE_INVALID;
+ uint32_t solid_chunk_size = UINT32_MAX;
+ int solid_ctype = WIMLIB_COMPRESSION_TYPE_INVALID;
int ret;
WIMStruct *wim;
const tchar *wimfile;
if (compression_type == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
- case IMAGEX_RECOMPRESS_OPTION:
+ case IMAGEX_COMPRESS_SLOW_OPTION:
+ set_compress_slow();
write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
break;
- case IMAGEX_COMPRESS_SLOW_OPTION:
+ case IMAGEX_RECOMPRESS_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
- set_compress_slow();
break;
case IMAGEX_CHUNK_SIZE_OPTION:
chunk_size = parse_chunk_size(optarg);
if (chunk_size == UINT32_MAX)
goto out_err;
break;
- case IMAGEX_PACK_CHUNK_SIZE_OPTION:
- pack_chunk_size = parse_chunk_size(optarg);
- if (pack_chunk_size == UINT32_MAX)
+ case IMAGEX_SOLID_CHUNK_SIZE_OPTION:
+ solid_chunk_size = parse_chunk_size(optarg);
+ if (solid_chunk_size == UINT32_MAX)
goto out_err;
break;
- case IMAGEX_PACK_COMPRESS_OPTION:
- pack_ctype = get_compression_type(optarg);
- if (pack_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
+ case IMAGEX_SOLID_COMPRESS_OPTION:
+ solid_ctype = get_compression_type(optarg);
+ if (solid_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
- case IMAGEX_PACK_STREAMS_OPTION:
+ case IMAGEX_SOLID_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_PACK_STREAMS;
write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
break;
if (ret)
goto out_wimlib_free;
}
- if (pack_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
- ret = wimlib_set_output_pack_compression_type(wim, pack_ctype);
+ if (solid_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
+ ret = wimlib_set_output_pack_compression_type(wim, solid_ctype);
if (ret)
goto out_wimlib_free;
}
- if (pack_chunk_size != UINT32_MAX) {
- ret = wimlib_set_output_pack_chunk_size(wim, pack_chunk_size);
+ if (solid_chunk_size != UINT32_MAX) {
+ ret = wimlib_set_output_pack_chunk_size(wim, solid_chunk_size);
if (ret)
goto out_wimlib_free;
}
goto out_free_command_str;
}
+/* Verify a WIM file. */
+static int
+imagex_verify(int argc, tchar **argv, int cmd)
+{
+ int ret;
+ const tchar *wimfile;
+ WIMStruct *wim;
+ int open_flags = WIMLIB_OPEN_FLAG_CHECK_INTEGRITY;
+ int verify_flags = 0;
+ STRING_SET(refglobs);
+ int c;
+
+ for_opt(c, verify_options) {
+ switch (c) {
+ case IMAGEX_REF_OPTION:
+ ret = string_set_append(&refglobs, optarg);
+ if (ret)
+ goto out_free_refglobs;
+ break;
+ case IMAGEX_NOCHECK_OPTION:
+ open_flags &= ~WIMLIB_OPEN_FLAG_CHECK_INTEGRITY;
+ break;
+ default:
+ goto out_usage;
+ }
+ }
+ argv += optind;
+ argc -= optind;
+
+ if (argc != 1) {
+ if (argc == 0)
+ imagex_error(T("Must specify a WIM file!"));
+ else
+ imagex_error(T("At most one WIM file can be specified!"));
+ goto out_usage;
+ }
+
+ wimfile = argv[0];
+
+ ret = wimlib_open_wim_with_progress(wimfile,
+ open_flags,
+ &wim,
+ imagex_progress_func,
+ NULL);
+ if (ret)
+ goto out_free_refglobs;
+
+ ret = wim_reference_globs(wim, &refglobs, open_flags);
+ if (ret)
+ goto out_wimlib_free;
+
+ ret = wimlib_verify_wim(wim, verify_flags);
+ if (ret) {
+ tputc(T('\n'), stderr);
+ imagex_error(T("\"%"TS"\" failed verification!"),
+ wimfile);
+ if (ret == WIMLIB_ERR_RESOURCE_NOT_FOUND &&
+ refglobs.num_strings == 0)
+ {
+ imagex_printf(T("Note: if this WIM file is not standalone, "
+ "use the --ref option to specify the other parts.\n"));
+ }
+ } else {
+ imagex_printf(T("\n\"%"TS"\" was successfully verified.\n"),
+ wimfile);
+ }
+
+out_wimlib_free:
+ wimlib_free(wim);
+out_free_refglobs:
+ string_set_destroy(&refglobs);
+ return ret;
+
+out_usage:
+ usage(CMD_VERIFY, stderr);
+ ret = -1;
+ goto out_free_refglobs;
+}
struct imagex_command {
const tchar *name;
[CMD_UNMOUNT] = {T("unmount"), imagex_unmount},
#endif
[CMD_UPDATE] = {T("update"), imagex_update},
+ [CMD_VERIFY] = {T("verify"), imagex_verify},
};
+#ifdef __WIN32__
+
+ /* Can be a directory or source list file. But source list file is probably
+ * a rare use case, so just say directory. */
+# define SOURCE_STR T("DIRECTORY")
+
+ /* Can only be a directory */
+# define TARGET_STR T("DIRECTORY")
+
+#else
+ /* Can be a directory, NTFS volume, or source list file. */
+# define SOURCE_STR T("SOURCE")
+
+ /* Can be a directory or NTFS volume. */
+# define TARGET_STR T("TARGET")
+
+#endif
+
static const tchar *usage_strings[] = {
[CMD_APPEND] =
T(
-" %"TS" (DIRECTORY | NTFS_VOLUME) WIMFILE\n"
-" [IMAGE_NAME [IMAGE_DESCRIPTION]] [--boot] [--check]\n"
-" [--nocheck] [--flags EDITION_ID] [--dereference]\n"
-" [--config=FILE] [--threads=NUM_THREADS] [--source-list]\n"
-" [--no-acls] [--strict-acls] [--rpfix] [--norpfix]\n"
-" [--update-of=[WIMFILE:]IMAGE] [--wimboot]\n"
+" %"TS" " SOURCE_STR " WIMFILE [IMAGE_NAME [IMAGE_DESC]]\n"
+" [--boot] [--check] [--nocheck] [--config=FILE]\n"
+" [--threads=NUM_THREADS] [--no-acls] [--strict-acls]\n"
+" [--rpfix] [--norpfix] [--update-of=[WIMFILE:]IMAGE]\n"
+" [--wimboot] [--unix-data] [--dereference]\n"
),
[CMD_APPLY] =
T(
-" %"TS" WIMFILE [(IMAGE_NUM | IMAGE_NAME | all)]\n"
-" (DIRECTORY | NTFS_VOLUME) [--check] [--ref=\"GLOB\"]\n"
-" [--no-acls] [--strict-acls] [--no-attributes]\n"
-" [--rpfix] [--norpfix] [--include-invalid-names]\n"
-" [--wimboot] [--unix-data]\n"
+" %"TS" WIMFILE [IMAGE] " TARGET_STR "\n"
+" [--check] [--ref=\"GLOB\"] [--no-acls] [--strict-acls]\n"
+" [--no-attributes] [--rpfix] [--norpfix]\n"
+" [--include-invalid-names] [--wimboot] [--unix-data]\n"
),
[CMD_CAPTURE] =
T(
-" %"TS" (DIRECTORY | NTFS_VOLUME) WIMFILE\n"
-" [IMAGE_NAME [IMAGE_DESCRIPTION]] [--boot] [--check]\n"
-" [--nocheck] [--compress=TYPE] [--flags EDITION_ID]\n"
-" [--dereference] [--config=FILE] [--threads=NUM_THREADS]\n"
-" [--source-list] [--no-acls] [--strict-acls] [--rpfix]\n"
-" [--norpfix] [--update-of=[WIMFILE:]IMAGE]\n"
-" [--delta-from=WIMFILE] [--wimboot] [--unix-data]\n"
+" %"TS" " SOURCE_STR " WIMFILE [IMAGE_NAME [IMAGE_DESC]]\n"
+" [--compress=TYPE] [--boot] [--check] [--nocheck]\n"
+" [--config=FILE] [--threads=NUM_THREADS]\n"
+" [--no-acls] [--strict-acls] [--rpfix] [--norpfix]\n"
+" [--update-of=[WIMFILE:]IMAGE] [--delta-from=WIMFILE]\n"
+" [--wimboot] [--unix-data] [--dereference] [--solid]\n"
),
[CMD_DELETE] =
T(
-" %"TS" WIMFILE (IMAGE_NUM | IMAGE_NAME | all)\n"
-" [--check] [--soft]\n"
+" %"TS" WIMFILE IMAGE [--check] [--soft]\n"
),
[CMD_DIR] =
T(
-" %"TS" WIMFILE (IMAGE_NUM | IMAGE_NAME | all) [--path=PATH] [--detailed]\n"
+" %"TS" WIMFILE IMAGE [--path=PATH] [--detailed]\n"
),
[CMD_EXPORT] =
T(
-" %"TS" SRC_WIMFILE (SRC_IMAGE_NUM | SRC_IMAGE_NAME | all ) \n"
-" DEST_WIMFILE [DEST_IMAGE_NAME [DEST_IMAGE_DESCRIPTION]]\n"
+" %"TS" SRC_WIMFILE SRC_IMAGE DEST_WIMFILE\n"
+" [DEST_IMAGE_NAME [DEST_IMAGE_DESC]]\n"
" [--boot] [--check] [--nocheck] [--compress=TYPE]\n"
" [--ref=\"GLOB\"] [--threads=NUM_THREADS] [--rebuild]\n"
" [--wimboot]\n"
),
[CMD_EXTRACT] =
T(
-" %"TS" WIMFILE (IMAGE_NUM | IMAGE_NAME) [(PATH | @LISTFILE)...]\n"
+" %"TS" WIMFILE IMAGE [(PATH | @LISTFILE)...]\n"
" [--check] [--ref=\"GLOB\"] [--dest-dir=CMD_DIR]\n"
" [--to-stdout] [--no-acls] [--strict-acls]\n"
" [--no-attributes] [--include-invalid-names]\n"
),
[CMD_INFO] =
T(
-" %"TS" WIMFILE [(IMAGE_NUM | IMAGE_NAME) [NEW_NAME\n"
-" [NEW_DESC]]] [--boot] [--check] [--nocheck] [--xml]\n"
+" %"TS" WIMFILE [IMAGE [NEW_NAME [NEW_DESC]]]\n"
+" [--boot] [--check] [--nocheck] [--xml]\n"
" [--extract-xml FILE] [--header] [--lookup-table]\n"
),
[CMD_JOIN] =
#if WIM_MOUNTING_SUPPORTED
[CMD_MOUNT] =
T(
-" %"TS" WIMFILE [(IMAGE_NUM | IMAGE_NAME)] DIRECTORY\n"
+" %"TS" WIMFILE [IMAGE] DIRECTORY\n"
" [--check] [--streams-interface=INTERFACE]\n"
" [--ref=\"GLOB\"] [--allow-other] [--unix-data]\n"
),
[CMD_MOUNTRW] =
T(
-" %"TS" WIMFILE [(IMAGE_NUM | IMAGE_NAME)] DIRECTORY\n"
+" %"TS" WIMFILE [IMAGE] DIRECTORY\n"
" [--check] [--streams-interface=INTERFACE]\n"
" [--staging-dir=CMD_DIR] [--allow-other] [--unix-data]\n"
),
#endif
[CMD_OPTIMIZE] =
T(
-" %"TS" WIMFILE [--check] [--nocheck] [--recompress]\n"
-" [--recompress-slow] [--compress=TYPE]\n"
-" [--threads=NUM_THREADS]\n"
+" %"TS" WIMFILE\n"
+" [--recompress] [--compress=TYPE]\n"
+" [--threads=NUM_THREADS] [--check] [--nocheck]\n"
+"\n"
),
[CMD_SPLIT] =
T(
#if WIM_MOUNTING_SUPPORTED
[CMD_UNMOUNT] =
T(
-" %"TS" DIRECTORY [--commit] [--force] [--new-image]\n"
-" [--check] [--rebuild]\n"
+" %"TS" DIRECTORY\n"
+" [--commit] [--force] [--new-image] [--check] [--rebuild]\n"
),
#endif
[CMD_UPDATE] =
T(
-" %"TS" WIMFILE [IMAGE_NUM | IMAGE_NAME] [--check] [--rebuild]\n"
-" [--threads=NUM_THREADS] [DEFAULT_ADD_OPTIONS]\n"
-" [DEFAULT_DELETE_OPTIONS] [--command=STRING]\n"
-" [--wimboot-config=FILE| [< CMDFILE]\n"
+" %"TS" WIMFILE [IMAGE]\n"
+" [--check] [--rebuild] [--threads=NUM_THREADS]\n"
+" [DEFAULT_ADD_OPTIONS] [DEFAULT_DELETE_OPTIONS]\n"
+" [--command=STRING] [--wimboot-config=FILE]\n"
+" [< CMDFILE]\n"
+),
+[CMD_VERIFY] =
+T(
+" %"TS" WIMFILE [--ref=\"GLOB\"]\n"
),
};
{
const tchar *format_str;
#ifdef __WIN32__
- format_str = T("Uncommon options are not listed;\n"
+ format_str = T("Some uncommon options are not listed;\n"
"See %"TS".pdf in the doc directory for more details.\n");
#else
- format_str = T("Uncommon options are not listed;\n"
+ format_str = T("Some uncommon options are not listed;\n"
"Try `man %"TS"' for more details.\n");
#endif
tfprintf(fp, format_str, get_cmd_string(cmd, true));
T(
" %"TS" --help\n"
" %"TS" --version\n"
-"\n"
-" The compression TYPE may be \"maximum\", \"fast\", or \"none\".\n"
"\n"
);
tfprintf(fp, extra, invocation_name, invocation_name);
+ tfprintf(fp,
+ T("IMAGE can be the 1-based index or name of an image in the WIM file.\n"
+ "For some commands IMAGE is optional if the WIM file only contains one image.\n"
+ "For some commands IMAGE may be \"all\".\n"
+ "\n"));
recommend_man_page(CMD_NONE, fp);
}