-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathkk_ihex_write.h
167 lines (151 loc) · 6.53 KB
/
kk_ihex_write.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
/*
* kk_ihex_write.h: A simple library for writing Intel HEX data.
* See the accompanying kk_ihex_read.h for read support, and the
* main header kk_ihex.h for the shared parts.
*
*
* 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 below 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
* -----------------
*
* If you are using only the write support, you should define
* `IHEX_LINE_MAX_LENGTH` as the length of your output line. This makes
* both the `struct ihex_state' and the internal write buffer smaller.
* For example, 32 or even 16 can be used instead of the default 255.
*
* If the write functionality is not used all the time and can thus
* share its write buffer memory with something else that is inactive
* during writing IHEX, you can define `IHEX_EXTERNAL_WRITE_BUFFER` and
* provide the buffer as `char *ihex_write_buffer`. The size of the
* buffer must be at least `IHEX_WRITE_BUFFER_LENGTH` bytes and it must
* be valid for the entire duration from the first call to a write function
* until after the last call to `ihex_end_write`. Note that there is
* no advantage to this unless something else, mutually exclusive with
* IHEX writing, can share the memory.
*
* If you are reading IHEX as well, then you'll end up limiting the
* maximum length of line that can be read. In that case you may wish to
* define `IHEX_MAX_OUTPUT_LINE_LENGTH` as smaller to decrease the
* write buffer size, but keep `IHEX_LINE_MAX_LENGTH` at 255 to support
* reading any IHEX file.
*
*
* 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_WRITE_H
#define KK_IHEX_WRITE_H
#ifdef __cplusplus
#ifndef restrict
#define restrict
#endif
extern "C" {
#else
#define restrict
#endif
#include "kk_ihex.h"
// Default number of data bytes written per line
#if IHEX_LINE_MAX_LENGTH >= 32
#define IHEX_DEFAULT_OUTPUT_LINE_LENGTH 32
#else
#define IHEX_DEFAULT_OUTPUT_LINE_LENGTH IHEX_LINE_MAX_LENGTH
#endif
#ifndef IHEX_MAX_OUTPUT_LINE_LENGTH
#define IHEX_MAX_OUTPUT_LINE_LENGTH IHEX_LINE_MAX_LENGTH
#endif
// Length of the write buffer required
#define IHEX_WRITE_BUFFER_LENGTH ( 1 + 2 + 4 + 2 + ( IHEX_MAX_OUTPUT_LINE_LENGTH * 2 ) + 2 + sizeof( IHEX_NEWLINE_STRING ) )
#ifdef IHEX_EXTERNAL_WRITE_BUFFER
// Define `IHEX_EXTERNAL_WRITE_BUFFER` to provide an external write buffer,
// as `char *ihex_write_buffer`, which must point to a valid storage for
// at least `IHEX_WRITE_BUFFER_LENGTH` characters whenever any of the
// write functionality is used (see above under "CONSERVING MEMORY").
extern char *ihex_write_buffer;
#endif
// Initialise the structure `ihex` for writing
void ihex_init( struct ihex_state *ihex );
// Begin writing at the given 32-bit `address` after writing any
// pending data at the current address.
//
// This can also be used to skip to a new address without calling
// `ihex_end_write`; this allows writing sparse output.
//
void ihex_write_at_address( struct ihex_state *ihex, ihex_address_t address );
// Write a single byte
void ihex_write_byte( struct ihex_state *ihex, int b );
// Write `count` bytes from `data`
void ihex_write_bytes( struct ihex_state *restrict ihex, const void *restrict data, ihex_count_t count );
// End writing (flush buffers, write end of file record)
void ihex_end_write( struct ihex_state *ihex );
// Called whenever the global, internal write buffer needs to be flushed by
// the write functions. The implementation is NOT provided by this library;
// this must be implemented to perform the actual output, i.e., write out
// `(eptr - buffer)` bytes from `buffer` (which is not NUL-terminated, but
// may be modified to make it thus).
//
// Example implementation:
//
// void ihex_flush_buffer(struct ihex_state *ihex,
// char *buffer, char *eptr) {
// *eptr = '\0';
// (void) fputs(buffer, stdout);
// }
//
// Note that the contents of `buffer` can become invalid immediately after
// this function returns - the data must be copied if it needs to be preserved!
//
extern void ihex_flush_buffer( struct ihex_state *ihex, char *buffer, char *eptr );
// As `ihex_write_at_address`, but specify a segment selector. Note that
// segments are not automatically incremented when the 16-bit address
// overflows (the default is to use 32-bit linear addressing). For segmented
// 20-bit addressing you must manually ensure that a write does not overflow
// the segment boundary, and call `ihex_write_at_segment` every time the
// segment needs to be changed.
//
#ifndef IHEX_DISABLE_SEGMENTS
void ihex_write_at_segment( struct ihex_state *ihex, ihex_segment_t segment, ihex_address_t address );
#endif
// Set the output line length to `length` - may be safely called only right
// after `ihex_write_at_address` or `ihex_write_at_segment`. The maximum
// is IHEX_LINE_MAX_LENGTH (which may be changed at compile time).
void ihex_set_output_line_length( struct ihex_state *ihex, uint8_t line_length );
#ifdef __cplusplus
}
#endif
#endif // !KK_IHEX_WRITE_H