123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- /*
- * mkdlinkfw
- *
- * Copyright (C) 2018 Paweł Dembicki <paweldembicki@gmail.com>
- *
- * This tool is based on mktplinkfw.
- * Copyright (C) 2009 Gabor Juhos <juhosg@openwrt.org>
- * Copyright (C) 2008,2009 Wang Jian <lark@linux.net.cn>
- *
- * 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 the Free
- * Software Foundation; either version 2 of the License, or (at your option)
- * any later version.
- */
- #include <stdio.h>
- #include <stdlib.h>
- #include <stdint.h>
- #include <string.h>
- #include <unistd.h> /* for unlink() */
- #include <libgen.h>
- #include <getopt.h> /* for getopt() */
- #include <stdarg.h>
- #include <stdbool.h>
- #include <endian.h>
- #include <errno.h>
- #include <time.h>
- #include <sys/stat.h>
- #include <zlib.h> /*for crc32 */
- #include "mkdlinkfw-lib.h"
- extern char *progname;
- uint32_t jboot_timestamp(void)
- {
- char *env = getenv("SOURCE_DATE_EPOCH");
- char *endptr = env;
- time_t fixed_timestamp = -1;
- errno = 0;
- if (env && *env) {
- fixed_timestamp = strtoull(env, &endptr, 10);
- if (errno || (endptr && *endptr != '\0')) {
- fprintf(stderr, "Invalid SOURCE_DATE_EPOCH");
- fixed_timestamp = -1;
- }
- }
- if (fixed_timestamp == -1)
- time(&fixed_timestamp);
- return (((uint32_t) fixed_timestamp) - TIMESTAMP_MAGIC) >> 2;
- }
- uint16_t jboot_checksum(uint16_t start_val, uint16_t *data, int size)
- {
- uint32_t counter = start_val;
- uint16_t *ptr = data;
- while (size > 1) {
- counter += *ptr;
- ++ptr;
- while (counter >> 16)
- counter = (uint16_t) counter + (counter >> 16);
- size -= 2;
- }
- if (size > 0) {
- counter += *(uint8_t *) ptr;
- counter -= 0xFF;
- }
- while (counter >> 16)
- counter = (uint16_t) counter + (counter >> 16);
- return counter;
- }
- int get_file_stat(struct file_info *fdata)
- {
- struct stat st;
- int res;
- if (fdata->file_name == NULL)
- return 0;
- res = stat(fdata->file_name, &st);
- if (res) {
- ERRS("stat failed on %s", fdata->file_name);
- return res;
- }
- fdata->file_size = st.st_size;
- return 0;
- }
- int read_to_buf(const struct file_info *fdata, char *buf)
- {
- FILE *f;
- int ret = EXIT_FAILURE;
- size_t read;
- f = fopen(fdata->file_name, "r");
- if (f == NULL) {
- ERRS("could not open \"%s\" for reading", fdata->file_name);
- goto out;
- }
- read = fread(buf, fdata->file_size, 1, f);
- if (ferror(f) || read != 1) {
- ERRS("unable to read from file \"%s\"", fdata->file_name);
- goto out_close;
- }
- ret = EXIT_SUCCESS;
- out_close:
- fclose(f);
- out:
- return ret;
- }
- int write_fw(const char *ofname, const char *data, int len)
- {
- FILE *f;
- int ret = EXIT_FAILURE;
- f = fopen(ofname, "w");
- if (f == NULL) {
- ERRS("could not open \"%s\" for writing", ofname);
- goto out;
- }
- errno = 0;
- fwrite(data, len, 1, f);
- if (errno) {
- ERRS("unable to write output file");
- goto out_flush;
- }
- DBG("firmware file \"%s\" completed", ofname);
- ret = EXIT_SUCCESS;
- out_flush:
- fflush(f);
- fclose(f);
- if (ret != EXIT_SUCCESS)
- unlink(ofname);
- out:
- return ret;
- }
|