Tom Rini | 10e4779 | 2018-05-06 17:58:06 -0400 | [diff] [blame] | 1 | // SPDX-License-Identifier: GPL-2.0+ |
Guilherme Maciel Ferreira | 8ed4d1c | 2013-12-01 12:43:10 -0700 | [diff] [blame] | 2 | /* |
| 3 | * (C) Copyright 2013 |
| 4 | * |
| 5 | * Written by Guilherme Maciel Ferreira <guilherme.maciel.ferreira@gmail.com> |
Guilherme Maciel Ferreira | 8ed4d1c | 2013-12-01 12:43:10 -0700 | [diff] [blame] | 6 | */ |
| 7 | |
| 8 | #include "imagetool.h" |
| 9 | |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 10 | #include <image.h> |
| 11 | |
Guilherme Maciel Ferreira | 28be1cf | 2015-01-15 02:48:07 -0200 | [diff] [blame] | 12 | struct image_type_params *imagetool_get_type(int type) |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 13 | { |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 14 | struct image_type_params **curr; |
| 15 | INIT_SECTION(image_type); |
| 16 | |
| 17 | struct image_type_params **start = __start_image_type; |
| 18 | struct image_type_params **end = __stop_image_type; |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 19 | |
Guilherme Maciel Ferreira | 28be1cf | 2015-01-15 02:48:07 -0200 | [diff] [blame] | 20 | for (curr = start; curr != end; curr++) { |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 21 | if ((*curr)->check_image_type) { |
| 22 | if (!(*curr)->check_image_type(type)) |
| 23 | return *curr; |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 24 | } |
| 25 | } |
| 26 | return NULL; |
| 27 | } |
| 28 | |
Pali Rohár | cdf0d1c | 2022-02-13 01:09:46 +0100 | [diff] [blame] | 29 | static int imagetool_verify_print_header_by_type( |
| 30 | void *ptr, |
| 31 | struct stat *sbuf, |
| 32 | struct image_type_params *tparams, |
| 33 | struct image_tool_params *params); |
| 34 | |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 35 | int imagetool_verify_print_header( |
| 36 | void *ptr, |
| 37 | struct stat *sbuf, |
| 38 | struct image_type_params *tparams, |
| 39 | struct image_tool_params *params) |
| 40 | { |
| 41 | int retval = -1; |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 42 | struct image_type_params **curr; |
| 43 | INIT_SECTION(image_type); |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 44 | |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 45 | struct image_type_params **start = __start_image_type; |
| 46 | struct image_type_params **end = __stop_image_type; |
Guilherme Maciel Ferreira | 28be1cf | 2015-01-15 02:48:07 -0200 | [diff] [blame] | 47 | |
Pali Rohár | cdf0d1c | 2022-02-13 01:09:46 +0100 | [diff] [blame] | 48 | if (tparams) |
| 49 | return imagetool_verify_print_header_by_type(ptr, sbuf, tparams, params); |
| 50 | |
Guilherme Maciel Ferreira | 28be1cf | 2015-01-15 02:48:07 -0200 | [diff] [blame] | 51 | for (curr = start; curr != end; curr++) { |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 52 | if ((*curr)->verify_header) { |
| 53 | retval = (*curr)->verify_header((unsigned char *)ptr, |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 54 | sbuf->st_size, params); |
| 55 | |
| 56 | if (retval == 0) { |
| 57 | /* |
Jordan Hand | 256b7f9 | 2019-03-05 14:47:56 -0800 | [diff] [blame] | 58 | * Print the image information if verify is |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 59 | * successful |
| 60 | */ |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 61 | if ((*curr)->print_header) { |
Simon Glass | da1805f | 2016-05-01 13:55:38 -0600 | [diff] [blame] | 62 | if (!params->quiet) |
| 63 | (*curr)->print_header(ptr); |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 64 | } else { |
| 65 | fprintf(stderr, |
| 66 | "%s: print_header undefined for %s\n", |
Andreas Bießmann | d631d70 | 2015-02-09 00:06:10 +0100 | [diff] [blame] | 67 | params->cmdname, (*curr)->name); |
Guilherme Maciel Ferreira | 0b33803 | 2015-01-15 02:48:05 -0200 | [diff] [blame] | 68 | } |
| 69 | break; |
| 70 | } |
| 71 | } |
| 72 | } |
| 73 | |
| 74 | return retval; |
| 75 | } |
Guilherme Maciel Ferreira | fc3c967 | 2015-01-15 02:48:06 -0200 | [diff] [blame] | 76 | |
Pali Rohár | cdf0d1c | 2022-02-13 01:09:46 +0100 | [diff] [blame] | 77 | static int imagetool_verify_print_header_by_type( |
Jordan Hand | 256b7f9 | 2019-03-05 14:47:56 -0800 | [diff] [blame] | 78 | void *ptr, |
| 79 | struct stat *sbuf, |
| 80 | struct image_type_params *tparams, |
| 81 | struct image_tool_params *params) |
| 82 | { |
| 83 | int retval; |
| 84 | |
| 85 | retval = tparams->verify_header((unsigned char *)ptr, sbuf->st_size, |
| 86 | params); |
| 87 | |
| 88 | if (retval == 0) { |
| 89 | /* |
| 90 | * Print the image information if verify is successful |
| 91 | */ |
| 92 | if (tparams->print_header) { |
| 93 | if (!params->quiet) |
| 94 | tparams->print_header(ptr); |
| 95 | } else { |
| 96 | fprintf(stderr, |
| 97 | "%s: print_header undefined for %s\n", |
| 98 | params->cmdname, tparams->name); |
| 99 | } |
| 100 | } else { |
| 101 | fprintf(stderr, |
| 102 | "%s: verify_header failed for %s with exit code %d\n", |
| 103 | params->cmdname, tparams->name, retval); |
| 104 | } |
| 105 | |
| 106 | return retval; |
| 107 | } |
| 108 | |
Guilherme Maciel Ferreira | f3f4a0a | 2015-01-15 02:54:41 -0200 | [diff] [blame] | 109 | int imagetool_save_subimage( |
Guilherme Maciel Ferreira | fc3c967 | 2015-01-15 02:48:06 -0200 | [diff] [blame] | 110 | const char *file_name, |
| 111 | ulong file_data, |
| 112 | ulong file_len) |
| 113 | { |
| 114 | int dfd; |
| 115 | |
| 116 | dfd = open(file_name, O_RDWR | O_CREAT | O_TRUNC | O_BINARY, |
| 117 | S_IRUSR | S_IWUSR); |
| 118 | if (dfd < 0) { |
| 119 | fprintf(stderr, "Can't open \"%s\": %s\n", |
| 120 | file_name, strerror(errno)); |
| 121 | return -1; |
| 122 | } |
| 123 | |
| 124 | if (write(dfd, (void *)file_data, file_len) != (ssize_t)file_len) { |
| 125 | fprintf(stderr, "Write error on \"%s\": %s\n", |
| 126 | file_name, strerror(errno)); |
| 127 | close(dfd); |
| 128 | return -1; |
| 129 | } |
| 130 | |
| 131 | close(dfd); |
| 132 | |
| 133 | return 0; |
| 134 | } |
Simon Glass | 1c57023 | 2016-02-22 22:55:49 -0700 | [diff] [blame] | 135 | |
| 136 | int imagetool_get_filesize(struct image_tool_params *params, const char *fname) |
| 137 | { |
| 138 | struct stat sbuf; |
| 139 | int fd; |
| 140 | |
| 141 | fd = open(fname, O_RDONLY | O_BINARY); |
| 142 | if (fd < 0) { |
| 143 | fprintf(stderr, "%s: Can't open %s: %s\n", |
| 144 | params->cmdname, fname, strerror(errno)); |
| 145 | return -1; |
| 146 | } |
| 147 | |
| 148 | if (fstat(fd, &sbuf) < 0) { |
| 149 | fprintf(stderr, "%s: Can't stat %s: %s\n", |
| 150 | params->cmdname, fname, strerror(errno)); |
Simon Glass | a1c090b | 2016-03-16 07:45:37 -0600 | [diff] [blame] | 151 | close(fd); |
Simon Glass | 1c57023 | 2016-02-22 22:55:49 -0700 | [diff] [blame] | 152 | return -1; |
| 153 | } |
| 154 | close(fd); |
| 155 | |
| 156 | return sbuf.st_size; |
| 157 | } |
Vagrant Cascadian | f8e066c | 2016-06-16 12:28:40 -0700 | [diff] [blame] | 158 | |
| 159 | time_t imagetool_get_source_date( |
Alex Kiernan | c4c7e0d | 2018-06-20 20:10:51 +0000 | [diff] [blame] | 160 | const char *cmdname, |
Vagrant Cascadian | f8e066c | 2016-06-16 12:28:40 -0700 | [diff] [blame] | 161 | time_t fallback) |
| 162 | { |
| 163 | char *source_date_epoch = getenv("SOURCE_DATE_EPOCH"); |
| 164 | |
| 165 | if (source_date_epoch == NULL) |
| 166 | return fallback; |
| 167 | |
| 168 | time_t time = (time_t) strtol(source_date_epoch, NULL, 10); |
| 169 | |
| 170 | if (gmtime(&time) == NULL) { |
| 171 | fprintf(stderr, "%s: SOURCE_DATE_EPOCH is not valid\n", |
Alex Kiernan | c4c7e0d | 2018-06-20 20:10:51 +0000 | [diff] [blame] | 172 | cmdname); |
Vagrant Cascadian | f8e066c | 2016-06-16 12:28:40 -0700 | [diff] [blame] | 173 | time = 0; |
| 174 | } |
| 175 | |
| 176 | return time; |
| 177 | } |