*/
/*
- * Copyright (C) 2012, 2013, 2014, 2015 Eric Biggers
+ * Copyright (C) 2012-2016 Eric Biggers
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
IMAGEX_COMMIT_OPTION,
IMAGEX_COMPACT_OPTION,
IMAGEX_COMPRESS_OPTION,
- IMAGEX_COMPRESS_SLOW_OPTION,
IMAGEX_CONFIG_OPTION,
IMAGEX_DEBUG_OPTION,
IMAGEX_DELTA_FROM_OPTION,
IMAGEX_FLAGS_OPTION,
IMAGEX_FORCE_OPTION,
IMAGEX_HEADER_OPTION,
+ IMAGEX_IMAGE_PROPERTY_OPTION,
IMAGEX_INCLUDE_INVALID_NAMES_OPTION,
IMAGEX_LAZY_OPTION,
IMAGEX_METADATA_OPTION,
IMAGEX_RECOMPRESS_OPTION,
IMAGEX_RECURSIVE_OPTION,
IMAGEX_REF_OPTION,
- IMAGEX_RESUME_OPTION,
IMAGEX_RPFIX_OPTION,
+ IMAGEX_SNAPSHOT_OPTION,
IMAGEX_SOFT_OPTION,
IMAGEX_SOLID_CHUNK_SIZE_OPTION,
IMAGEX_SOLID_COMPRESS_OPTION,
{T("rpfix"), no_argument, NULL, IMAGEX_RPFIX_OPTION},
{T("norpfix"), no_argument, NULL, IMAGEX_NORPFIX_OPTION},
{T("include-invalid-names"), no_argument, NULL, IMAGEX_INCLUDE_INVALID_NAMES_OPTION},
-
- /* --resume is undocumented for now as it needs improvement. */
- {T("resume"), no_argument, NULL, IMAGEX_RESUME_OPTION},
{T("wimboot"), no_argument, NULL, IMAGEX_WIMBOOT_OPTION},
{T("compact"), required_argument, NULL, IMAGEX_COMPACT_OPTION},
{NULL, 0, NULL, 0},
{T("no-check"), no_argument, NULL, IMAGEX_NOCHECK_OPTION},
{T("nocheck"), no_argument, NULL, IMAGEX_NOCHECK_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("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("no-solid-sort"), no_argument, NULL, IMAGEX_NO_SOLID_SORT_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("image-property"), required_argument, NULL, IMAGEX_IMAGE_PROPERTY_OPTION},
{T("verbose"), no_argument, NULL, IMAGEX_VERBOSE_OPTION},
{T("threads"), required_argument, NULL, IMAGEX_THREADS_OPTION},
{T("rebuild"), no_argument, NULL, IMAGEX_REBUILD_OPTION},
{T("delta-from"), required_argument, NULL, IMAGEX_DELTA_FROM_OPTION},
{T("wimboot"), no_argument, NULL, IMAGEX_WIMBOOT_OPTION},
{T("unsafe-compact"), no_argument, NULL, IMAGEX_UNSAFE_COMPACT_OPTION},
+ {T("snapshot"), no_argument, NULL, IMAGEX_SNAPSHOT_OPTION},
{NULL, 0, NULL, 0},
};
{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("chunk-size"), required_argument, NULL, IMAGEX_CHUNK_SIZE_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("no-solid-sort"), no_argument, NULL, IMAGEX_NO_SOLID_SORT_OPTION},
{T("ref"), required_argument, NULL, IMAGEX_REF_OPTION},
{T("threads"), required_argument, NULL, IMAGEX_THREADS_OPTION},
{T("header"), no_argument, NULL, IMAGEX_HEADER_OPTION},
{T("lookup-table"), no_argument, NULL, IMAGEX_BLOBS_OPTION},
{T("blobs"), no_argument, NULL, IMAGEX_BLOBS_OPTION},
- {T("metadata"), no_argument, NULL, IMAGEX_METADATA_OPTION},
{T("xml"), no_argument, NULL, IMAGEX_XML_OPTION},
+ {T("image-property"), required_argument, NULL, IMAGEX_IMAGE_PROPERTY_OPTION},
{NULL, 0, NULL, 0},
};
{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("recompress-slow"), no_argument, NULL, IMAGEX_COMPRESS_SLOW_OPTION},
{T("chunk-size"), required_argument, NULL, IMAGEX_CHUNK_SIZE_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("no-solid-sort"), no_argument, NULL, IMAGEX_NO_SOLID_SORT_OPTION},
{T("threads"), required_argument, NULL, IMAGEX_THREADS_OPTION},
{T("pipable"), no_argument, NULL, IMAGEX_PIPABLE_OPTION},
tfputs(s, fp);
}
-/* Parse the argument to --compress */
+/* Parse the argument to --compress or --solid-compress */
static int
-get_compression_type(tchar *optarg)
+get_compression_type(tchar *optarg, bool solid)
{
int ctype;
unsigned int compression_level = 0;
if (!tstrcasecmp(optarg, T("maximum")) ||
!tstrcasecmp(optarg, T("lzx")) ||
- !tstrcasecmp(optarg, T("max")))
+ !tstrcasecmp(optarg, T("max"))) {
ctype = WIMLIB_COMPRESSION_TYPE_LZX;
- else if (!tstrcasecmp(optarg, T("fast")) || !tstrcasecmp(optarg, T("xpress")))
+ } else if (!tstrcasecmp(optarg, T("fast")) || !tstrcasecmp(optarg, T("xpress"))) {
ctype = WIMLIB_COMPRESSION_TYPE_XPRESS;
- else if (!tstrcasecmp(optarg, T("recovery")) || !tstrcasecmp(optarg, T("lzms")))
+ } else if (!tstrcasecmp(optarg, T("recovery"))) {
+ if (!solid) {
+ tfprintf(stderr,
+T(
+"Warning: use of '--compress=recovery' is discouraged because it behaves\n"
+" differently from DISM. Instead, you typically want to use '--solid' to\n"
+" create a solid LZMS-compressed WIM or \"ESD file\", similar to DISM's\n"
+" /compress:recovery. But if you really want *non-solid* LZMS compression,\n"
+" then you may suppress this warning by specifying '--compress=lzms' instead\n"
+" of '--compress=recovery'.\n"));
+ }
ctype = WIMLIB_COMPRESSION_TYPE_LZMS;
- else if (!tstrcasecmp(optarg, T("none")))
+ } else if (!tstrcasecmp(optarg, T("lzms"))) {
+ ctype = WIMLIB_COMPRESSION_TYPE_LZMS;
+ } else if (!tstrcasecmp(optarg, T("none"))) {
ctype = WIMLIB_COMPRESSION_TYPE_NONE;
- else {
+ } else {
imagex_error(T("Invalid compression type \"%"TS"\"!"), optarg);
print_available_compression_types(stderr);
return WIMLIB_COMPRESSION_TYPE_INVALID;
}
-static void
-set_compress_slow(void)
-{
-#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 {
- const tchar **strings;
+ tchar **strings;
unsigned num_strings;
unsigned num_alloc_strings;
};
struct string_set _strings = STRING_SET_INITIALIZER
static int
-string_set_append(struct string_set *set, const tchar *glob)
+string_set_append(struct string_set *set, tchar *glob)
{
unsigned num_alloc_strings = set->num_alloc_strings;
if (set->num_strings == num_alloc_strings) {
- const tchar **new_strings;
+ tchar **new_strings;
num_alloc_strings += 4;
new_strings = realloc(set->strings,
static int
wim_reference_globs(WIMStruct *wim, struct string_set *set, int open_flags)
{
- return wimlib_reference_resource_files(wim, set->strings,
+ return wimlib_reference_resource_files(wim, (const tchar **)set->strings,
set->num_strings,
WIMLIB_REF_FLAG_GLOB_ENABLE,
open_flags);
}
+static int
+append_image_property_argument(struct string_set *image_properties)
+{
+ if (!tstrchr(optarg, '=')) {
+ imagex_error(T("'--image-property' argument "
+ "must be in the form NAME=VALUE"));
+ return -1;
+ }
+ return string_set_append(image_properties, optarg);
+}
+
+static int
+apply_image_properties(struct string_set *image_properties,
+ WIMStruct *wim, int image, bool *any_changes_ret)
+{
+ bool any_changes = false;
+ for (unsigned i = 0; i < image_properties->num_strings; i++) {
+ tchar *name, *value;
+ const tchar *current_value;
+ int ret;
+
+ name = image_properties->strings[i];
+ value = tstrchr(name, '=');
+ *value++ = '\0';
+
+ current_value = wimlib_get_image_property(wim, image, name);
+ if (current_value && !tstrcmp(current_value, value)) {
+ imagex_printf(T("The %"TS" property of image %d "
+ "already has value \"%"TS"\".\n"),
+ name, image, value);
+ } else {
+ imagex_printf(T("Setting the %"TS" property of image "
+ "%d to \"%"TS"\".\n"),
+ name, image, value);
+ ret = wimlib_set_image_property(wim, image, name, value);
+ if (ret)
+ return ret;
+ any_changes = true;
+ }
+ }
+ if (any_changes_ret)
+ *any_changes_ret = any_changes;
+ return 0;
+}
+
static void
do_resource_not_found_warning(const tchar *wimfile,
const struct wimlib_wim_info *info,
switch (msg) {
case WIMLIB_PROGRESS_MSG_WRITE_STREAMS:
{
- static bool first = true;
- if (first) {
- imagex_printf(T("Writing %"TS"-compressed data "
- "using %u thread%"TS"\n"),
- wimlib_get_compression_type_string(
- info->write_streams.compression_type),
- info->write_streams.num_threads,
- (info->write_streams.num_threads == 1) ? T("") : T("s"));
- first = false;
+ static bool started;
+ if (!started) {
+ if (info->write_streams.compression_type != WIMLIB_COMPRESSION_TYPE_NONE) {
+ imagex_printf(T("Using %"TS" compression "
+ "with %u thread%"TS"\n"),
+ wimlib_get_compression_type_string(
+ info->write_streams.compression_type),
+ info->write_streams.num_threads,
+ (info->write_streams.num_threads == 1) ? T("") : T("s"));
+ }
+ started = true;
}
}
unit_shift = get_unit(info->write_streams.total_bytes, &unit_name);
percent_done = TO_PERCENT(info->write_streams.completed_bytes,
info->write_streams.total_bytes);
- imagex_printf(T("\r%"PRIu64" %"TS" of %"PRIu64" %"TS" (uncompressed) "
- "written (%u%% done)"),
+ imagex_printf(T("\rArchiving file data: %"PRIu64" %"TS" of %"PRIu64" %"TS" (%u%%) done"),
info->write_streams.completed_bytes >> unit_shift,
unit_name,
info->write_streams.total_bytes >> unit_shift,
extract_flags |= WIMLIB_EXTRACT_FLAG_REPLACE_INVALID_FILENAMES;
extract_flags |= WIMLIB_EXTRACT_FLAG_ALL_CASE_CONFLICTS;
break;
- case IMAGEX_RESUME_OPTION:
- extract_flags |= WIMLIB_EXTRACT_FLAG_RESUME;
- break;
case IMAGEX_WIMBOOT_OPTION:
extract_flags |= WIMLIB_EXTRACT_FLAG_WIMBOOT;
break;
int open_flags = 0;
int add_flags = WIMLIB_ADD_FLAG_EXCLUDE_VERBOSE |
WIMLIB_ADD_FLAG_WINCONFIG |
- WIMLIB_ADD_FLAG_VERBOSE;
+ WIMLIB_ADD_FLAG_VERBOSE |
+ WIMLIB_ADD_FLAG_FILE_PATHS_UNNEEDED;
int write_flags = 0;
int compression_type = WIMLIB_COMPRESSION_TYPE_INVALID;
uint32_t chunk_size = UINT32_MAX;
const tchar *wimfile;
int wim_fd;
const tchar *name;
- const tchar *desc;
- const tchar *flags_element = NULL;
+ STRING_SET(image_properties);
WIMStruct *wim;
STRING_SET(base_wimfiles);
add_flags &= ~WIMLIB_ADD_FLAG_WINCONFIG;
break;
case IMAGEX_COMPRESS_OPTION:
- compression_type = get_compression_type(optarg);
+ compression_type = get_compression_type(optarg, false);
if (compression_type == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
- case IMAGEX_COMPRESS_SLOW_OPTION:
- 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_SOLID_COMPRESS_OPTION:
- solid_ctype = get_compression_type(optarg);
+ solid_ctype = get_compression_type(optarg, true);
if (solid_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
case IMAGEX_NO_SOLID_SORT_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_NO_SOLID_SORT;
break;
- case IMAGEX_FLAGS_OPTION:
- flags_element = optarg;
+ case IMAGEX_FLAGS_OPTION: {
+ tchar *p = alloca((6 + tstrlen(optarg) + 1) * sizeof(tchar));
+ tsprintf(p, T("FLAGS=%"TS), optarg);
+ ret = string_set_append(&image_properties, p);
+ if (ret)
+ goto out;
+ break;
+ }
+ case IMAGEX_IMAGE_PROPERTY_OPTION:
+ ret = append_image_property_argument(&image_properties);
+ if (ret)
+ goto out;
break;
case IMAGEX_DEREFERENCE_OPTION:
add_flags |= WIMLIB_ADD_FLAG_DEREFERENCE;
}
ret = string_set_append(&base_wimfiles, optarg);
if (ret)
- goto out_free_base_wimfiles;
+ goto out;
write_flags |= WIMLIB_WRITE_FLAG_SKIP_EXTERNAL_WIMS;
break;
case IMAGEX_WIMBOOT_OPTION:
}
write_flags |= WIMLIB_WRITE_FLAG_UNSAFE_COMPACT;
break;
+ case IMAGEX_SNAPSHOT_OPTION:
+ add_flags |= WIMLIB_ADD_FLAG_SNAPSHOT;
+ break;
default:
goto out_usage;
}
name = tbasename(tstrcpy(source_copy, source));
name_defaulted = true;
}
- /* Image description defaults to NULL if not given. */
- if (argc >= 4)
- desc = argv[3];
- else
- desc = NULL;
+
+ /* Image description (if given). */
+ if (argc >= 4) {
+ tchar *p = alloca((12 + tstrlen(argv[3]) + 1) * sizeof(tchar));
+ tsprintf(p, T("DESCRIPTION=%"TS), argv[3]);
+ ret = string_set_append(&image_properties, p);
+ if (ret)
+ goto out;
+ }
if (source_list) {
/* Set up capture sources in source list mode */
ret = wimlib_set_output_chunk_size(wim, chunk_size);
if (ret)
goto out_free_wim;
- } else if ((add_flags & WIMLIB_ADD_FLAG_WIMBOOT) &&
- compression_type == WIMLIB_COMPRESSION_TYPE_XPRESS) {
- ret = wimlib_set_output_chunk_size(wim, 4096);
- if (ret)
- goto out_free_wim;
+ } else if ((add_flags & WIMLIB_ADD_FLAG_WIMBOOT)) {
+
+ int ctype = compression_type;
+
+ if (cmd == CMD_APPEND) {
+ struct wimlib_wim_info info;
+ wimlib_get_wim_info(wim, &info);
+ ctype = info.compression_type;
+ }
+
+ if (ctype == WIMLIB_COMPRESSION_TYPE_XPRESS) {
+ ret = wimlib_set_output_chunk_size(wim, 4096);
+ if (ret)
+ goto out_free_wim;
+ }
}
if (solid_ctype != WIMLIB_COMPRESSION_TYPE_INVALID) {
ret = wimlib_set_output_pack_compression_type(wim, solid_ctype);
if (ret)
goto out_free_template_wim;
- if (desc || flags_element || template_image_name_or_num) {
- /* User provided <DESCRIPTION> or <FLAGS> element, or an image
- * on which the added one is to be based has been specified with
- * --update-of. Get the index of the image we just
- * added, then use it to call the appropriate functions. */
+ if (image_properties.num_strings || template_image_name_or_num) {
+ /* User asked to set additional image properties, or an image on
+ * which the added one is to be based has been specified with
+ * --update-of. */
struct wimlib_wim_info info;
wimlib_get_wim_info(wim, &info);
- if (desc) {
- ret = wimlib_set_image_descripton(wim,
- info.image_count,
- desc);
- if (ret)
- goto out_free_template_wim;
- }
-
- if (flags_element) {
- ret = wimlib_set_image_flags(wim, info.image_count,
- flags_element);
- if (ret)
- goto out_free_template_wim;
- }
+ ret = apply_image_properties(&image_properties, wim,
+ info.image_count, NULL);
+ if (ret)
+ goto out_free_template_wim;
/* Reference template image if the user provided one. */
if (template_image_name_or_num) {
free(capture_sources);
out_free_source_list_contents:
free(source_list_contents);
-out_free_base_wimfiles:
+out:
+ string_set_destroy(&image_properties);
string_set_destroy(&base_wimfiles);
return ret;
usage(cmd, stderr);
out_err:
ret = -1;
- goto out_free_base_wimfiles;
+ goto out;
}
/* Remove image(s) from a WIM. */
wimlib_iterate_lookup_table(wim, 0, print_resource, NULL);
}
+#ifndef __WIN32__
static void
default_print_security_descriptor(const uint8_t *sd, size_t size)
{
print_byte_field(sd, size);
tputchar(T('\n'));
}
+#endif
static void
print_dentry_detailed(const struct wimlib_dir_entry *dentry)
write_flags |= WIMLIB_WRITE_FLAG_NO_CHECK_INTEGRITY;
break;
case IMAGEX_COMPRESS_OPTION:
- compression_type = get_compression_type(optarg);
+ compression_type = get_compression_type(optarg, false);
if (compression_type == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
- case IMAGEX_COMPRESS_SLOW_OPTION:
- set_compress_slow();
- write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
- break;
case IMAGEX_RECOMPRESS_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
break;
goto out_err;
break;
case IMAGEX_SOLID_COMPRESS_OPTION:
- solid_ctype = get_compression_type(optarg);
+ solid_ctype = get_compression_type(optarg, true);
if (solid_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
const tchar *xml_out_file = NULL;
const tchar *wimfile;
const tchar *image_num_or_name;
- const tchar *new_name;
- const tchar *new_desc;
+ STRING_SET(image_properties);
WIMStruct *wim;
int image;
int ret;
xml_out_file = optarg;
short_header = false;
break;
- case IMAGEX_METADATA_OPTION:
- imagex_error(T("The --metadata option has been removed. "
- "Use 'wimdir --detail' instead."));
- goto out_err;
+ case IMAGEX_IMAGE_PROPERTY_OPTION:
+ ret = append_image_property_argument(&image_properties);
+ if (ret)
+ goto out;
+ break;
default:
goto out_usage;
}
wimfile = argv[0];
image_num_or_name = (argc >= 2) ? argv[1] : T("all");
- new_name = (argc >= 3) ? argv[2] : NULL;
- new_desc = (argc >= 4) ? argv[3] : NULL;
+
+ if (argc >= 3) {
+ /* NEW_NAME */
+ tchar *p = alloca((5 + tstrlen(argv[2]) + 1) * sizeof(tchar));
+ tsprintf(p, T("NAME=%"TS), argv[2]);
+ ret = string_set_append(&image_properties, p);
+ if (ret)
+ goto out;
+ }
+
+ if (argc >= 4) {
+ /* NEW_DESC */
+ tchar *p = alloca((12 + tstrlen(argv[3]) + 1) * sizeof(tchar));
+ tsprintf(p, T("DESCRIPTION=%"TS), argv[3]);
+ ret = string_set_append(&image_properties, p);
+ if (ret)
+ goto out;
+ }
if (check && nocheck) {
imagex_error(T("Can't specify both --check and --nocheck"));
"image in a multi-image WIM"));
goto out_wimlib_free;
}
- if (new_name) {
- imagex_error(T("Cannot specify the NEW_NAME "
- "without specifying a specific "
- "image in a multi-image WIM"));
+ if (image_properties.num_strings) {
+ imagex_error(T("Can't change image properties without "
+ "specifying a specific image in a "
+ "multi-image WIM"));
goto out_wimlib_free;
}
}
/* Operations that print information are separated from operations that
* recreate the WIM file. */
- if (!new_name && !boot) {
+ if (!image_properties.num_strings && !boot) {
/* Read-only operations */
ret = 0;
} else {
-
/* Modification operations */
+ bool any_property_changes;
if (image == WIMLIB_ALL_IMAGES)
image = 1;
- if (image == WIMLIB_NO_IMAGE && new_name) {
- imagex_error(T("Cannot specify new_name (\"%"TS"\") "
- "when using image 0"), new_name);
+ if (image == WIMLIB_NO_IMAGE && image_properties.num_strings) {
+ imagex_error(T("Cannot change image properties "
+ "when using image 0"));
ret = -1;
goto out_wimlib_free;
}
goto out_wimlib_free;
}
}
- if (new_name) {
- if (!tstrcmp(wimlib_get_image_name(wim, image), new_name))
- {
- imagex_printf(T("Image %d is already named \"%"TS"\".\n"),
- image, new_name);
- new_name = NULL;
- } else {
- imagex_printf(T("Changing the name of image %d to "
- "\"%"TS"\".\n"), image, new_name);
- ret = wimlib_set_image_name(wim, image, new_name);
- if (ret)
- goto out_wimlib_free;
- }
- }
- if (new_desc) {
- const tchar *old_desc;
- old_desc = wimlib_get_image_description(wim, image);
- if (old_desc && !tstrcmp(old_desc, new_desc)) {
- imagex_printf(T("The description of image %d is already "
- "\"%"TS"\".\n"), image, new_desc);
- new_desc = NULL;
- } else {
- imagex_printf(T("Changing the description of image %d "
- "to \"%"TS"\".\n"), image, new_desc);
- ret = wimlib_set_image_descripton(wim, image,
- new_desc);
- if (ret)
- goto out_wimlib_free;
- }
- }
+
+ ret = apply_image_properties(&image_properties, wim, image,
+ &any_property_changes);
+ if (ret)
+ goto out_wimlib_free;
/* Only call wimlib_overwrite() if something actually needs to
* be changed. */
- if (boot || new_name || new_desc ||
+ if (boot || any_property_changes ||
(check && !info.has_integrity_table) ||
(nocheck && info.has_integrity_table))
{
out_wimlib_free:
wimlib_free(wim);
out:
+ string_set_destroy(&image_properties);
return ret;
out_usage:
break;
case IMAGEX_COMPRESS_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
- compression_type = get_compression_type(optarg);
+ compression_type = get_compression_type(optarg, false);
if (compression_type == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
- case IMAGEX_COMPRESS_SLOW_OPTION:
- set_compress_slow();
- write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
- break;
case IMAGEX_RECOMPRESS_OPTION:
write_flags |= WIMLIB_WRITE_FLAG_RECOMPRESS;
break;
goto out_err;
break;
case IMAGEX_SOLID_COMPRESS_OPTION:
- solid_ctype = get_compression_type(optarg);
+ solid_ctype = get_compression_type(optarg, true);
if (solid_ctype == WIMLIB_COMPRESSION_TYPE_INVALID)
goto out_err;
break;
" [--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"
+" [--wimboot] [--unix-data] [--dereference] [--snapshot]\n"
),
[CMD_APPLY] =
T(
" [--no-acls] [--strict-acls] [--rpfix] [--norpfix]\n"
" [--update-of=[WIMFILE:]IMAGE] [--delta-from=WIMFILE]\n"
" [--wimboot] [--unix-data] [--dereference] [--solid]\n"
+" [--snapshot]\n"
),
[CMD_DELETE] =
T(
" %"TS" WIMFILE [IMAGE [NEW_NAME [NEW_DESC]]]\n"
" [--boot] [--check] [--nocheck] [--xml]\n"
" [--extract-xml FILE] [--header] [--blobs]\n"
+" [--image-property NAME=VALUE]\n"
),
[CMD_JOIN] =
T(
static const tchar *s =
T(
"wimlib-imagex (distributed with " PACKAGE " " PACKAGE_VERSION ")\n"
-"Copyright (C) 2012, 2013, 2014, 2015 Eric Biggers\n"
+"Copyright (C) 2012-2016 Eric Biggers\n"
"License GPLv3+; GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>.\n"
"This is free software: you are free to change and redistribute it.\n"
"There is NO WARRANTY, to the extent permitted by law.\n"
recommend_man_page(CMD_NONE, fp);
}
+#ifdef __WIN32__
+extern int wmain(int argc, wchar_t **argv);
+#define main wmain
+#endif
+
/* Entry point for wimlib's ImageX implementation. On UNIX the command
* arguments will just be 'char' strings (ideally UTF-8 encoded, but could be
* something else), while on Windows the command arguments will be UTF-16LE
* encoded 'wchar_t' strings. */
int
-#ifdef __WIN32__
-wmain(int argc, wchar_t **argv, wchar_t **envp)
-#else
-main(int argc, char **argv)
-#endif
+main(int argc, tchar **argv)
{
int ret;
int init_flags = 0;