Browse Source

init

master
heck 2 years ago
commit
14e7cf69fe
  1. 42
      .clang-format
  2. 4
      .gitignore
  3. 20
      LICENSE
  4. 0
      README.md
  5. 1025
      data/flangers.hex
  6. BIN
      data/flangers.hex.bin.target
  7. 45
      src/Makefile
  8. 194
      src/fvhex2bin.c
  9. 191
      src/kk_ihex.h
  10. 181
      src/kk_ihex_read.c
  11. 117
      src/kk_ihex_read.h

42
.clang-format

@ -0,0 +1,42 @@
BasedOnStyle: LLVM
ReflowComments: false
MacroBlockBegin: "^BEGIN_OPERATOR"
MacroBlockEnd: "^END_OPERATOR"
Language: Cpp
DerivePointerAlignment: true
SortIncludes: Never
PointerAlignment: Left
AlignAfterOpenBracket: AlwaysBreak
AlignOperands: AlignAfterOperator
AlignTrailingComments: true
AllowAllArgumentsOnNextLine: false
AllowAllParametersOfDeclarationOnNextLine: false
AllowShortEnumsOnASingleLine: false
AllowShortFunctionsOnASingleLine: Empty
AllowShortIfStatementsOnASingleLine: Never
AllowShortLoopsOnASingleLine: false
BinPackArguments: false
BinPackParameters: false
ExperimentalAutoDetectBinPacking: true
BreakBeforeBraces: Custom
BraceWrapping:
AfterFunction: true
ColumnLimit: 100
AlwaysBreakAfterDefinitionReturnType: None
AlwaysBreakAfterReturnType: None
PenaltyBreakBeforeFirstCallParameter: 0
PenaltyReturnTypeOnItsOwnLine: 1000000
PenaltyBreakAssignment: 1000000
PenaltyExcessCharacter: 10
IndentCaseLabels: true
IndentWidth: 4
MaxEmptyLinesToKeep: 2
NamespaceIndentation: All
SpaceAfterTemplateKeyword: false
AccessModifierOffset: -4
AllowShortBlocksOnASingleLine: Always
IndentPPDirectives: BeforeHash
IndentExternBlock: Indent
Cpp11BracedListStyle: false

4
.gitignore

@ -0,0 +1,4 @@
*.o
*.d
src/fvhex2bin
src/libfvhex2bin.a

20
LICENSE

@ -0,0 +1,20 @@
The MIT License (MIT)
Copyright (c) 2013-2017 Kimmo Kulovesi
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

0
README.md

1025
data/flangers.hex

File diff suppressed because it is too large

BIN
data/flangers.hex.bin.target

Binary file not shown.

45
src/Makefile

@ -0,0 +1,45 @@
TARGET_EXE=fvhex2bin
TARGET_LIB=lib$(TARGET_EXE).a
# default config
#SYS_PREFIX?=/opt/local
# System
#SYS_INC_PATH?=$(SYS_PREFIX)/include
#SYS_LIB_PATH?=$(SYS_PREFIX)/lib
CFLAGS+=-std=c99 -MMD -MP -Wno-format-extra-args -Wno-deprecated-declarations -fvisibility=hidden
#CFLAGS+=-isystem$(SYS_INC_PATH)
#LDFLAGS+=-DMACOSX_DEPLOYMENT_TARGET=11.00 -framework Foundation -framework IOKit -framework Security
#STATIC_LINK_LIBS=$(SYS_LIB_PATH)/libusb-1.0.a
SRC_ALL=$(wildcard *.c)
SRC_EXE=$(TARGET_EXE).c
SRC_LIB=$(filter-out $(SRC_EXE), $(SRC_ALL))
#OBJ_EXE=$(subst .c,.o,$(SRC_EXE))
OBJ_LIB=$(subst .c,.o,$(SRC_LIB))
DEPENDS=$(subst .c,.d,$(SRC_ALL))
ifneq ($(MAKECMDGOALS),clean)
-include $(DEPENDS)
endif
.PHONY: all clean uninstall install
.DEFAULT_GOAL := all
all: $(TARGET_EXE)
lib: $(TARGET_LIB)
$(TARGET_EXE): $(SRC_EXE) $(TARGET_LIB)
$(CXX) $(LDFLAGS) $(TARGET_LIB) $(STATIC_LINK_LIBS) $(SRC_EXE) -o $@
$(TARGET_LIB): $(OBJ_LIB)
$(AR) -cr $@ $^
clean:
rm $(OBJ_LIB) $(TARGET_LIB) $(TARGET_EXE) $(DEPENDS)

194
src/fvhex2bin.c

@ -0,0 +1,194 @@
/*
* ihex2bin.c: Read Intel HEX format, write binary data.
*
* By default reads from stdin and writes to stdout. The command-line
* options `-i` and `-o` can be used to specify the input and output
* file, respectively. Specifying an output file allows sparse writes.
*
* NOTE: Many Intel HEX files produced by compilers/etc have data
* beginning at an address greater than zero, potentially causing very
* unnecessarily large files to be created. The command-line option
* `-a` can be used to specify the start address of the output file,
* i.e., the value will be subtracted from the IHEX addresses (the
* result must not be negative).
*
* Alternatively, the command-line option `-A` sets the address offset
* to the first address that would be written (i.e., first byte of
* data written will be at address 0).
*
* Copyright (c) 2013-2019 Kimmo Kulovesi, https://arkku.com
* Provided with absolutely no warranty, use at your own risk only.
* Distribute freely, mark modified copies as such.
*/
#include "kk_ihex_read.h"
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#define AUTODETECT_ADDRESS (~0UL)
static FILE *outfile;
static unsigned long line_number = 1L;
static unsigned long file_position = 0L;
static unsigned long address_offset = 0UL;
static bool debug_enabled = 0;
int main(int argc, char *argv[])
{
struct ihex_state ihex;
FILE *infile = stdin;
ihex_count_t count;
char buf[256];
outfile = stdout;
while (--argc) {
char *arg = *(++argv);
if (arg[0] == '-' && arg[1] && arg[2] == '\0') {
switch (arg[1]) {
case 'a':
if (--argc == 0) {
goto invalid_argument;
}
++argv;
errno = 0;
address_offset = strtoul(*argv, &arg, 0);
if (errno || arg == *argv) {
errno = errno ? errno : EINVAL;
goto argument_error;
}
break;
case 'A':
address_offset = AUTODETECT_ADDRESS;
break;
case 'i':
if (--argc == 0) {
goto invalid_argument;
}
++argv;
if (!(infile = fopen(*argv, "r"))) {
goto argument_error;
}
break;
case 'o':
if (--argc == 0) {
goto invalid_argument;
}
++argv;
if (!(outfile = fopen(*argv, "wb"))) {
goto argument_error;
}
break;
case 'v':
debug_enabled = 1;
break;
case 'h':
case '?':
arg = NULL;
goto usage;
default:
goto invalid_argument;
}
continue;
}
invalid_argument:
(void)fprintf(stderr, "Invalid argument: %s\n", arg);
usage:
(void)fprintf(stderr, "kk_ihex " KK_IHEX_VERSION " - Copyright (c) 2013-2015 Kimmo Kulovesi\n");
(void)fprintf(
stderr,
"Usage: ihex2bin ([-a <address_offset>]|[-A])"
" [-o <out.bin>] [-i <in.hex>] [-v]\n");
return arg ? EXIT_FAILURE : EXIT_SUCCESS;
argument_error:
perror(*argv);
return EXIT_FAILURE;
}
ihex_read_at_address(
&ihex,
(address_offset != AUTODETECT_ADDRESS) ? (ihex_address_t)address_offset : 0);
while (fgets(buf, sizeof(buf), infile)) {
count = (ihex_count_t)strlen(buf);
ihex_read_bytes(&ihex, buf, count);
line_number += (count && buf[count - 1] == '\n');
}
ihex_end_read(&ihex);
if (infile != stdin) {
(void)fclose(infile);
}
return EXIT_SUCCESS;
}
ihex_bool_t ihex_data_read(struct ihex_state *ihex, ihex_record_type_t type, ihex_bool_t error)
{
if (error) {
(void)fprintf(stderr, "Checksum error on line %lu\n", line_number);
exit(EXIT_FAILURE);
}
if ((error = (ihex->length < ihex->line_length))) {
(void)fprintf(stderr, "Line length error on line %lu\n", line_number);
exit(EXIT_FAILURE);
}
if (!outfile) {
(void)fprintf(stderr, "Excess data after end of file record\n");
exit(EXIT_FAILURE);
}
if (type == IHEX_DATA_RECORD) {
unsigned long address = (unsigned long)IHEX_LINEAR_ADDRESS(ihex);
if (address < address_offset) {
if (address_offset == AUTODETECT_ADDRESS) {
// autodetect initial address
address_offset = address;
if (debug_enabled) {
(void)fprintf(stderr, "Address offset: 0x%lx\n", address_offset);
}
} else {
(void)fprintf(stderr, "Address underflow on line %lu\n", line_number);
exit(EXIT_FAILURE);
}
}
address -= address_offset;
if (address != file_position) {
if (debug_enabled) {
(void)fprintf(
stderr,
"Seeking from 0x%lx to 0x%lx on line %lu\n",
file_position,
address,
line_number);
}
if (outfile == stdout || fseek(outfile, (long)address, SEEK_SET)) {
if (file_position < address) {
// "seek" forward in stdout by writing NUL bytes
do {
(void)fputc('\0', outfile);
} while (++file_position < address);
} else {
perror("fseek");
exit(EXIT_FAILURE);
}
}
file_position = address;
}
if (!fwrite(ihex->data, ihex->length, 1, outfile)) {
perror("fwrite");
exit(EXIT_FAILURE);
}
file_position += ihex->length;
} else if (type == IHEX_END_OF_FILE_RECORD) {
if (debug_enabled) {
(void)fprintf(stderr, "%lu bytes written\n", file_position);
}
if (outfile != stdout) {
(void)fclose(outfile);
}
outfile = NULL;
}
return true;
}

191
src/kk_ihex.h

@ -0,0 +1,191 @@
/*
* kk_ihex.h: A simple library for reading and writing the Intel HEX
* or IHEX format. Intended mainly for embedded systems, and thus
* somewhat optimised for size at the expense of error handling and
* generality.
*
* USAGE
* -----
*
* The library has been split into read and write parts, which use a
* common data structure (`struct ihex_state`), but each can be used
* independently. Include the header `kk_ihex_read.h` for reading, and/or
* the header `kk_ihex_write.h` for writing (and link with their respective
* object files). Both can be used simultaneously - this header defines
* the shared data structures and definitions.
*
*
* READING INTEL HEX DATA
* ----------------------
*
* To read data in the Intel HEX format, you must perform the actual reading
* of bytes using other means (e.g., stdio). The bytes read must then be
* passed to `ihex_read_byte` and/or `ihex_read_bytes`. The reading functions
* will then call `ihex_data_read`, at which stage the `struct ihex_state`
* structure will contain the data along with its address. See the header
* `kk_ihex_read.h` for details and example implementation of `ihex_data_read`.
*
* The sequence to read data in IHEX format is:
* struct ihex_state ihex;
* ihex_begin_read(&ihex);
* ihex_read_bytes(&ihex, my_input_bytes, length_of_my_input_bytes);
* ihex_end_read(&ihex);
*
*
* WRITING BINARY DATA AS INTEL HEX
* --------------------------------
*
* In order to write out data, the `ihex_write_at_address` or
* `ihex_write_at_segment` functions are used to set the data location,
* and then the binary bytes are written with `ihex_write_byte` and/or
* `ihex_write_bytes`. The writing functions will then call the function
* `ihex_flush_buffer` whenever the internal write buffer needs to be
* cleared - it is up to the caller to provide an implementation of
* `ihex_flush_buffer` to do the actual writing. See the header
* `kk_ihex_write.h` for details and an example implementation.
*
* See the declaration further down for an example implementation.
*
* The sequence to write data in IHEX format is:
* struct ihex_state ihex;
* ihex_init(&ihex);
* ihex_write_at_address(&ihex, 0);
* ihex_write_bytes(&ihex, my_data, length_of_my_data);
* ihex_end_write(&ihex);
*
* For outputs larger than 64KiB, 32-bit linear addresses are output. Normally
* the initial linear extended address record of zero is NOT written - it can
* be forced by setting `ihex->flags |= IHEX_FLAG_ADDRESS_OVERFLOW` before
* writing the first byte.
*
* Gaps in the data may be created by calling `ihex_write_at_address` with the
* new starting address without calling `ihex_end_write` in between.
*
*
* The same `struct ihex_state` may be used either for reading or writing,
* but NOT both at the same time. Furthermore, a global output buffer is
* used for writing, i.e., multiple threads must not write simultaneously
* (but multiple writes may be interleaved).
*
*
* CONSERVING MEMORY
* -----------------
*
* For memory-critical use, you can save additional memory by defining
* `IHEX_LINE_MAX_LENGTH` as something less than 255. Note, however, that
* this limit affects both reading and writing, so the resulting library
* will be unable to read lines with more than this number of data bytes.
* That said, I haven't encountered any IHEX files with more than 32
* data bytes per line. For write only there is no reason to define the
* maximum as greater than the line length you'll actually be writing,
* e.g., 32 or 16.
*
* If the write functionality is only occasionally used, you can provide
* your own buffer for the duration by defining `IHEX_EXTERNAL_WRITE_BUFFER`
* and providing a `char *ihex_write_buffer` which points to valid storage
* for at least `IHEX_WRITE_BUFFER_LENGTH` characters from before the first
* call to any IHEX write function to until after the last.
*
* If you are doing both reading and writing, you can define the maximum
* output length separately as `IHEX_MAX_OUTPUT_LINE_LENGTH` - this will
* decrease the write buffer size, but `struct ihex_state` will still
* use the larger `IHEX_LINE_MAX_LENGTH` for its data storage.
*
* You can also save a few additional bytes by disabling support for
* segmented addresses, by defining `IHEX_DISABLE_SEGMENTS`. Both the
* read and write modules need to be build with the same option, as the
* resulting data structures will not be compatible otherwise. To be honest,
* this is a fairly pointless optimisation.
*
*
* Copyright (c) 2013-2019 Kimmo Kulovesi, https://arkku.com/
* Provided with absolutely no warranty, use at your own risk only.
* Use and distribute freely, mark modified copies as such.
*/
#ifndef KK_IHEX_H
#define KK_IHEX_H
#define KK_IHEX_VERSION "2019-08-07"
#include <stdint.h>
#ifdef IHEX_USE_STDBOOL
#include <stdbool.h>
typedef bool ihex_bool_t;
#else
typedef uint_fast8_t ihex_bool_t;
#endif
typedef uint_least32_t ihex_address_t;
typedef uint_least16_t ihex_segment_t;
typedef int ihex_count_t;
// Maximum number of data bytes per line (applies to both reading and
// writing!); specify 255 to support reading all possible lengths. Less
// can be used to limit memory footprint on embedded systems, e.g.,
// most programs with IHEX output use 32.
#ifndef IHEX_LINE_MAX_LENGTH
#define IHEX_LINE_MAX_LENGTH 255
#endif
enum ihex_flags {
IHEX_FLAG_ADDRESS_OVERFLOW = 0x80 // 16-bit address overflow
};
typedef uint8_t ihex_flags_t;
typedef struct ihex_state {
ihex_address_t address;
#ifndef IHEX_DISABLE_SEGMENTS
ihex_segment_t segment;
#endif
ihex_flags_t flags;
uint8_t line_length;
uint8_t length;
uint8_t data[IHEX_LINE_MAX_LENGTH + 1];
} kk_ihex_t;
enum ihex_record_type {
IHEX_DATA_RECORD,
IHEX_END_OF_FILE_RECORD,
IHEX_EXTENDED_SEGMENT_ADDRESS_RECORD,
IHEX_START_SEGMENT_ADDRESS_RECORD,
IHEX_EXTENDED_LINEAR_ADDRESS_RECORD,
IHEX_START_LINEAR_ADDRESS_RECORD
};
typedef uint8_t ihex_record_type_t;
#ifndef IHEX_DISABLE_SEGMENTS
// Resolve segmented address (if any). It is the author's recommendation that
// segmented addressing not be used (and indeed the write function of this
// library uses linear 32-bit addressing unless manually overridden).
//
#define IHEX_LINEAR_ADDRESS(ihex) ((ihex)->address + (((ihex_address_t)((ihex)->segment)) << 4))
//
// Note that segmented addressing with the above macro is not strictly adherent
// to the IHEX specification, which mandates that the lowest 16 bits of the
// address and the index of the data byte must be added modulo 64K (i.e.,
// at 16 bits precision with wraparound) and the segment address only added
// afterwards.
//
// To implement fully correct segmented addressing, compute the address
// of _each byte_ with its index in `data` as follows:
//
#define IHEX_BYTE_ADDRESS(ihex, byte_index) ((((ihex)->address + (byte_index)) & 0xFFFFU) + (((ihex_address_t)((ihex)->segment)) << 4))
#else // IHEX_DISABLE_SEGMENTS:
#define IHEX_LINEAR_ADDRESS(ihex) ((ihex)->address)
#define IHEX_BYTE_ADDRESS(ihex, byte_index) ((ihex)->address + (byte_index))
#endif
// The newline string (appended to every output line, e.g., "\r\n")
#ifndef IHEX_NEWLINE_STRING
#define IHEX_NEWLINE_STRING "\n"
#endif
// See kk_ihex_read.h and kk_ihex_write.h for function declarations!
#endif // !KK_IHEX_H

181
src/kk_ihex_read.c

@ -0,0 +1,181 @@
/*
* kk_ihex_read.c: A simple library for reading the Intel HEX (IHEX) format.
*
* See the header `kk_ihex.h` for instructions.
*
* Copyright (c) 2013-2019 Kimmo Kulovesi, https://arkku.com/
* Provided with absolutely no warranty, use at your own risk only.
* Use and distribute freely, mark modified copies as such.
*/
#include "kk_ihex_read.h"
#define IHEX_START ':'
#define ADDRESS_HIGH_MASK ((ihex_address_t) 0xFFFF0000U)
enum ihex_read_state {
READ_WAIT_FOR_START = 0,
READ_COUNT_HIGH = 1,
READ_COUNT_LOW,
READ_ADDRESS_MSB_HIGH,
READ_ADDRESS_MSB_LOW,
READ_ADDRESS_LSB_HIGH,
READ_ADDRESS_LSB_LOW,
READ_RECORD_TYPE_HIGH,
READ_RECORD_TYPE_LOW,
READ_DATA_HIGH,
READ_DATA_LOW
};
#define IHEX_READ_RECORD_TYPE_MASK 0x07
#define IHEX_READ_STATE_MASK 0x78
#define IHEX_READ_STATE_OFFSET 3
void
ihex_begin_read (struct ihex_state * const ihex) {
ihex->address = 0;
#ifndef IHEX_DISABLE_SEGMENTS
ihex->segment = 0;
#endif
ihex->flags = 0;
ihex->line_length = 0;
ihex->length = 0;
}
void
ihex_read_at_address (struct ihex_state * const ihex, ihex_address_t address) {
ihex_begin_read(ihex);
ihex->address = address;
}
#ifndef IHEX_DISABLE_SEGMENTS
void
ihex_read_at_segment (struct ihex_state * const ihex, ihex_segment_t segment) {
ihex_begin_read(ihex);
ihex->segment = segment;
}
#endif
void
ihex_end_read (struct ihex_state * const ihex) {
uint_fast8_t type = ihex->flags & IHEX_READ_RECORD_TYPE_MASK;
uint_fast8_t sum;
if ((sum = ihex->length) == 0 && type == IHEX_DATA_RECORD) {
return;
}
{
// compute and validate checksum
const uint8_t * const eptr = ihex->data + sum;
const uint8_t *r = ihex->data;
sum += type + (ihex->address & 0xFFU) + ((ihex->address >> 8) & 0xFFU);
while (r != eptr) {
sum += *r++;
}
sum = (~sum + 1U) ^ *eptr; // *eptr is the received checksum
}
if (ihex_data_read(ihex, type, (uint8_t) sum)) {
if (type == IHEX_EXTENDED_LINEAR_ADDRESS_RECORD) {
ihex->address &= 0xFFFFU;
ihex->address |= (((ihex_address_t) ihex->data[0]) << 24) |
(((ihex_address_t) ihex->data[1]) << 16);
#ifndef IHEX_DISABLE_SEGMENTS
} else if (type == IHEX_EXTENDED_SEGMENT_ADDRESS_RECORD) {
ihex->segment = (ihex_segment_t) ((ihex->data[0] << 8) | ihex->data[1]);
#endif
}
}
ihex->length = 0;
ihex->flags = 0;
}
void
ihex_read_byte (struct ihex_state * const ihex, const char byte) {
uint_fast8_t b = (uint_fast8_t) byte;
uint_fast8_t len = ihex->length;
uint_fast8_t state = (ihex->flags & IHEX_READ_STATE_MASK);
ihex->flags ^= state; // turn off the old state
state >>= IHEX_READ_STATE_OFFSET;
if (b >= '0' && b <= '9') {
b -= '0';
} else if (b >= 'A' && b <= 'F') {
b -= 'A' - 10;
} else if (b >= 'a' && b <= 'f') {
b -= 'a' - 10;
} else if (b == IHEX_START) {
// sync to a new record at any state
state = READ_COUNT_HIGH;
goto end_read;
} else {
// ignore unknown characters (e.g., extra whitespace)
goto save_read_state;
}
if (!(++state & 1)) {
// high nybble, store temporarily at end of data:
b <<= 4;
ihex->data[len] = b;
} else {
// low nybble, combine with stored high nybble:
b = (ihex->data[len] |= b);
// We already know the lowest bit of `state`, dropping it may produce
// smaller code, hence the `>> 1` in switch and its cases.
switch (state >> 1) {
default:
// remain in initial state while waiting for :
return;
case (READ_COUNT_LOW >> 1):
// data length
ihex->line_length = b;
#if IHEX_LINE_MAX_LENGTH < 255
if (b > IHEX_LINE_MAX_LENGTH) {
ihex_end_read(ihex);
return;
}
#endif
break;
case (READ_ADDRESS_MSB_LOW >> 1):
// high byte of 16-bit address
ihex->address &= ADDRESS_HIGH_MASK; // clear the 16-bit address
ihex->address |= ((ihex_address_t) b) << 8U;
break;
case (READ_ADDRESS_LSB_LOW >> 1):
// low byte of 16-bit address
ihex->address |= (ihex_address_t) b;
break;
case (READ_RECORD_TYPE_LOW >> 1):
// record type
if (b & ~IHEX_READ_RECORD_TYPE_MASK) {
// skip unknown record types silently
return;
}
ihex->flags = (ihex->flags & ~IHEX_READ_RECORD_TYPE_MASK) | b;
break;
case (READ_DATA_LOW >> 1):
if (len < ihex->line_length) {
// data byte
ihex->length = len + 1;
state = READ_DATA_HIGH;
goto save_read_state;
}
// end of line (last "data" byte is checksum)
state = READ_WAIT_FOR_START;
end_read:
ihex_end_read(ihex);
}
}
save_read_state:
ihex->flags |= state << IHEX_READ_STATE_OFFSET;
}
void
ihex_read_bytes (struct ihex_state * restrict ihex,
const char * restrict data,
ihex_count_t count) {
while (count > 0) {
ihex_read_byte(ihex, *data++);
--count;
}
}

117
src/kk_ihex_read.h

@ -0,0 +1,117 @@
/*
* kk_ihex_read.h: A simple library for reading Intel HEX data. See
* the accompanying kk_ihex_write.h for IHEX write support.
*
*
* READING INTEL HEX DATA
* ----------------------
*
* To read data in the Intel HEX format, you must perform the actual reading
* of bytes using other means (e.g., stdio). The bytes read must then be
* passed to `ihex_read_byte` and/or `ihex_read_bytes`. The reading functions
* will then call `ihex_data_read`, at which stage the `struct ihex_state`
* structure will contain the data along with its address. See below for
* details and example implementation of `ihex_data_read`.
*
* The sequence to read data in IHEX format is:
* struct ihex_state ihex;
* ihex_begin_read(&ihex);
* ihex_read_bytes(&ihex, my_input_bytes, length_of_my_input_bytes);
* ihex_end_read(&ihex);
*
*
* CONSERVING MEMORY
* -----------------
*
* For memory-critical use, you can save additional memory by defining
* `IHEX_LINE_MAX_LENGTH` as something less than 255. Note, however, that
* this limit affects both reading and writing, so the resulting library
* will be unable to read lines with more than this number of data bytes.
* That said, I haven't encountered any IHEX files with more than 32
* data bytes per line.
*
*
* Copyright (c) 2013-2019 Kimmo Kulovesi, https://arkku.com/
* Provided with absolutely no warranty, use at your own risk only.
* Use and distribute freely, mark modified copies as such.
*/
#ifndef KK_IHEX_READ_H
#define KK_IHEX_READ_H
#ifdef __cplusplus
#ifndef restrict
#define restrict
#endif
extern "C" {
#endif
#include "kk_ihex.h"
// Begin reading at address 0
void ihex_begin_read(struct ihex_state *ihex);
// Begin reading at `address` (the lowest 16 bits of which will be ignored);
// this is required only if the high bytes of the 32-bit starting address
// are not specified in the input data and they are non-zero
void ihex_read_at_address(struct ihex_state *ihex,
ihex_address_t address);
// Read a single character
void ihex_read_byte(struct ihex_state *ihex, char chr);
// Read `count` bytes from `data`
void ihex_read_bytes(struct ihex_state * restrict ihex,
const char * restrict data,
ihex_count_t count);
// End reading (may call `ihex_data_read` if there is data waiting)
void ihex_end_read(struct ihex_state *ihex);
// Called when a complete line has been read, the record type of which is
// passed as `type`. The `ihex` structure will have its fields `data`,
// `line_length`, `address`, and `segment` set appropriately. In case
// of reading an `IHEX_EXTENDED_LINEAR_ADDRESS_RECORD` or an
// `IHEX_EXTENDED_SEGMENT_ADDRESS_RECORD` the record's data is not
// yet parsed - it will be parsed into the `address` or `segment` field
// only if `ihex_data_read` returns `true`. This allows manual handling
// of extended addresses by parsing the `ihex->data` bytes.
//
// Possible error cases include checksum mismatch (which is indicated
// as an argument), and excessive line length (in case this has been
// compiled with `IHEX_LINE_MAX_LENGTH` less than 255) which is indicated
// by `line_length` greater than `length`. Unknown record types and
// other erroneous data is usually silently ignored by this minimalistic
// parser. (It is recommended to compute a hash over the complete data
// once received and verify that against the source.)
//
// Example implementation:
//
// ihex_bool_t ihex_data_read(struct ihex_state *ihex,
// ihex_record_type_t type,
// ihex_bool_t error) {
// error = error || (ihex->length < ihex->line_length);
// if (type == IHEX_DATA_RECORD && !error) {
// (void) fseek(outfile, IHEX_LINEAR_ADDRESS(ihex), SEEK_SET);
// (void) fwrite(ihex->data, 1, ihex->length, outfile);
// } else if (type == IHEX_END_OF_FILE_RECORD) {
// (void) fclose(outfile);
// }
// return !error;
// }
//
extern ihex_bool_t ihex_data_read(struct ihex_state *ihex,
ihex_record_type_t type,
ihex_bool_t checksum_mismatch);
// Begin reading at `segment`; this is required only if the initial segment
// is not specified in the input data and it is non-zero.
//
#ifndef IHEX_DISABLE_SEGMENTS
void ihex_read_at_segment(struct ihex_state *ihex, ihex_segment_t segment);
#endif
#ifdef __cplusplus
}
#endif
#endif // !KK_IHEX_READ_H
Loading…
Cancel
Save