2018-02-03 11:59:50 +00:00
|
|
|
/*
|
|
|
|
* 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)
|
|
|
|
{
|
2019-01-05 06:34:02 +00:00
|
|
|
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;
|
2018-02-03 11:59:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
2019-01-05 06:40:55 +00:00
|
|
|
size_t read;
|
2018-02-03 11:59:50 +00:00
|
|
|
|
|
|
|
f = fopen(fdata->file_name, "r");
|
|
|
|
if (f == NULL) {
|
|
|
|
ERRS("could not open \"%s\" for reading", fdata->file_name);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-01-05 06:40:55 +00:00
|
|
|
read = fread(buf, fdata->file_size, 1, f);
|
|
|
|
if (ferror(f) || read != 1) {
|
2018-02-03 11:59:50 +00:00
|
|
|
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;
|
|
|
|
}
|